This file is indexed.

/usr/include/libindi/alignment/ConvexHull.h is in libindi-dev 1.7.1-0ubuntu1.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
/*!
 * \file ConvexHull.h
 *
 * \author Roger James
 * \date December 2013
 *
 */

#pragma once

// This C++ code is based on the c code described below
// it was ported to c++ by Roger James in December 2013
// !!!!!!!!!!!!!!!!!!!!!!! IMPORTANT !!!!!!!!!!!!!!!!!!
// This must code must use integer coordinates. A naive conversion
// to floating point WILL NOT work. For the reasons behind this
// have a look at at section 4.3.5 of the O'Rourke book. For more
// information try http://www.mpi-inf.mpg.de/departments/d1/ClassroomExamples/
// For INDI alignment purposes we need to scale floating point coordinates
// into the integer space before using this algorithm.

/*
This code is described in "Computational Geometry in C" (Second Edition),
Chapter 4.  It is not written to be comprehensible without the
explanation in that book.

Input: 3n integer coordinates for the points.
Output: the 3D convex hull, in postscript with embedded comments
        showing the vertices and faces.

Compile: gcc -o chull chull.c (or simply: make)

Written by Joseph O'Rourke, with contributions by
  Kristy Anderson, John Kutcher, Catherine Schevon, Susan Weller.
Last modified: May 2000
Questions to orourke@cs.smith.edu.

--------------------------------------------------------------------
This code is Copyright 2000 by Joseph O'Rourke.  It may be freely
redistributed in its entirety provided that this copyright notice is
not removed.
--------------------------------------------------------------------
*/

#include <gsl/gsl_matrix.h>

#include <cmath>
#include <cstring>
#include <fstream>
#include <limits>

namespace INDI
{
namespace AlignmentSubsystem
{
/*!
 * \class ConvexHull
 * \brief This class computes the convex hull of a set of 3d points.
 */
class ConvexHull
{
  public:
    ConvexHull();
    virtual ~ConvexHull() {}

    enum
    {
        X = 0,
        Y = 1,
        Z = 2
    };

    template <class Type>
    static void add(Type &head, Type p)
    {
        if (NULL != head)
        {
            p->next       = head;
            p->prev       = head->prev;
            head->prev    = p;
            p->prev->next = p;
        }
        else
        {
            head       = p;
            head->next = head->prev = p;
        }
    };

    template <class Type>
    static void remove(Type &head, Type p)
    {
        if (NULL != head)
        {
            if (head == head->next)
                head = NULL;
            else if (p == head)
                head = head->next;
            p->next->prev = p->prev;
            p->prev->next = p->next;
            delete p;
        }
    };

    template <class Type>
    static void swap(Type &t, Type &x, Type &y)
    {
        t = x;
        x = y;
        y = t;
    };

    // Explicit forward declarations
    struct tVertexStructure;
    struct tFaceStructure;
    struct tEdgeStructure;

    /* Define structures for vertices, edges and faces */
    typedef struct tVertexStructure tsVertex;
    typedef tsVertex *tVertex;

    typedef struct tEdgeStructure tsEdge;
    typedef tsEdge *tEdge;

    typedef struct tFaceStructure tsFace;
    typedef tsFace *tFace;

    struct tVertexStructure
    {
        int v[3];
        int vnum;
        tEdge duplicate; // pointer to incident cone edge (or NULL)
        bool onhull;     // True iff point on hull.
        bool mark;       // True iff point already processed.
        tVertex next, prev;
    };

    struct tEdgeStructure
    {
        tFace adjface[2];
        tVertex endpts[2];
        tFace newface;  // pointer to incident cone face.
        bool delete_it; //  True iff edge should be delete.
        tEdge next, prev;
    };

    struct tFaceStructure
    {
        tFaceStructure() { pMatrix = gsl_matrix_alloc(3, 3); }
        ~tFaceStructure() { gsl_matrix_free(pMatrix); }
        tEdge edge[3];
        tVertex vertex[3];
        bool visible; // True iff face visible from new point.
        tFace next, prev;
        gsl_matrix *pMatrix;
    };

    /* Define flags */
    static const bool ONHULL    = true;
    static const bool REMOVED   = true;
    static const bool VISIBLE   = true;
    static const bool PROCESSED = true;
    static const int SAFE       = 1000000; /* Range of safe coord values. */

    tVertex vertices;
    tEdge edges;
    tFace faces;

    /** \brief AddOne is passed a vertex.  It first determines all faces visible from
        that point.  If none are visible then the point is marked as not
        onhull.  Next is a loop over edges.  If both faces adjacent to an edge
        are visible, then the edge is marked for deletion.  If just one of the
        adjacent faces is visible then a new face is constructed.
        */
    bool AddOne(tVertex p);

    /** \brief Checks that, for each face, for each i={0,1,2}, the [i]th vertex of
        that face is either the [0]th or [1]st endpoint of the [ith] edge of
        the face.
        */
    void CheckEndpts();

    /** \brief CheckEuler checks Euler's relation, as well as its implications when
        all faces are known to be triangles.  Only prints positive information
        when debug is true, but always prints negative information.
        */
    void CheckEuler(int V, int E, int F);

    /** \brief Checks the consistency of the hull and prints the results to the
        standard error output.
        */
    void Checks();

    /** \brief CleanEdges runs through the edge list and cleans up the structure.
        If there is a newface then it will put that face in place of the
        visible face and NULL out newface. It also deletes so marked edges.
        */
    void CleanEdges();

    /** \brief CleanFaces runs through the face list and deletes any face marked visible.
        */
    void CleanFaces();

    /** \brief CleanUp goes through each data structure list and clears all
        flags and NULLs out some pointers.  The order of processing
        (edges, faces, vertices) is important.
        */
    void CleanUp(tVertex *pvnext);

    /** \brief CleanVertices runs through the vertex list and deletes the
        vertices that are marked as processed but are not incident to any
        undeleted edges.
        The pointer to vnext, pvnext, is used to alter vnext in
        ConstructHull() if we are about to delete vnext.
        */
    void CleanVertices(tVertex *pvnext);

    /** \brief Collinear checks to see if the three points given are collinear,
        by checking to see if each element of the cross product is zero.
        */
    bool Collinear(tVertex a, tVertex b, tVertex c);

    /** \brief Consistency runs through the edge list and checks that all
        adjacent faces have their endpoints in opposite order.  This verifies
        that the vertices are in counterclockwise order.
        */
    void Consistency();

    /** \brief ConstructHull adds the vertices to the hull one at a time.  The hull
        vertices are those in the list marked as onhull.
        */
    void ConstructHull();

    /** \brief Convexity checks that the volume between every face and every
        point is negative.  This shows that each point is inside every face
        and therefore the hull is convex.
        */
    void Convexity();

    /** \brief DoubleTriangle builds the initial double triangle.  It first finds 3
         noncollinear points and makes two faces out of them, in opposite order.
         It then finds a fourth point that is not coplanar with that face.  The
         vertices are stored in the face structure in counterclockwise order so
         that the volume between the face and the point is negative. Lastly, the
         3 newfaces to the fourth point are constructed and the data structures
         are cleaned up.
        */
    void DoubleTriangle();

    /** \brief EdgeOrderOnFaces: puts e0 between v0 and v1, e1 between v1 and v2,
          e2 between v2 and v0 on each face.  This should be unnecessary, alas.
          Not used in code, but useful for other purposes.
        */
    void EdgeOrderOnFaces();

    /** \brief Set the floating point to integer scaling factor
        */
    int GetScaleFactor() const { return ScaleFactor; }

    /** \brief MakeCcw puts the vertices in the face structure in counterclock wise
        order.  We want to store the vertices in the same
        order as in the visible face.  The third vertex is always p. Although no
        specific ordering of the edges of a face are used
        by the code, the following condition is maintained for each face f:
        one of the two endpoints of f->edge[i] matches f->vertex[i].
        But note that this does not imply that f->edge[i] is between
        f->vertex[i] and f->vertex[(i+1)%3].  (Thanks to Bob Williamson.)
        */
    void MakeCcw(tFace f, tEdge e, tVertex p);

    /** \brief MakeConeFace makes a new face and two new edges between the
        edge and the point that are passed to it. It returns a pointer to
        the new face.
        */
    tFace MakeConeFace(tEdge e, tVertex p);

    /** \brief MakeFace creates a new face structure from three vertices (in ccw
        order).  It returns a pointer to the face.
        */
    tFace MakeFace(tVertex v0, tVertex v1, tVertex v2, tFace f);

    /** \brief Makes a vertex from the supplied information and adds it to the vertices list.
        */
    void MakeNewVertex(double x, double y, double z, int VertexId);

    /** \brief MakeNullEdge creates a new cell and initializes all pointers to NULL
        and sets all flags to off.  It returns a pointer to the empty cell.
        */
    tEdge MakeNullEdge();

    /** \brief MakeNullFace creates a new face structure and initializes all of its
        flags to NULL and sets all the flags to off.  It returns a pointer
        to the empty cell.
        */
    tFace MakeNullFace();

    /** \brief MakeNullVertex: Makes a vertex, nulls out fields.
        */
    tVertex MakeNullVertex();

    /** \brief Print: Prints out the vertices and the faces.  Uses the vnum indices
        corresponding to the order in which the vertices were input.
        Output is in PostScript format. This code ignores the Z component of all vertices
        and does not scale the output to fit the page. It use on 3D hulls is not
        recommended.
        */
    void Print();

    /** \brief Prints the edges Ofile
        */
    void PrintEdges(std::ofstream &Ofile);

    /** \brief Prints the faces to Ofile
        */
    void PrintFaces(std::ofstream &Ofile);

    /** \brief Outputs the faces in Lightwave obj format for 3d viewing.
        The files chull.obj and chull.mtl are written to the current working
        directory.
        */
    void PrintObj(const char *FileName = "chull.obj");

    /** \brief Prints vertices, edges and faces to the standard error output
        */
    void PrintOut(const char *FileName, tVertex v);

    /** \brief Prints a single vertex to the standard output.
        */
    void PrintPoint(tVertex p);

    /** \brief Prints vertices to Ofile.
        */
    void PrintVertices(std::ofstream &Ofile);

    /** \brief ReadVertices: Reads in the vertices, and links them into a circular
        list with MakeNullVertex.  There is no need for the # of vertices to be
        the first line: the function looks for EOF instead.  Sets the global
        variable vertices via the add<> template function.
        */
    void ReadVertices();

    /** \brief Frees the vertices edges and faces lists and resets the debug and check flags.
        */
    void Reset();

    /** \brief Set the floating point to integer scaling factor. If you want to tweak
        this a good value to start from may well be a little bit more than the resolution of the
        mounts encoders. Whatever is used must not exceed the default value which is
        set to the constant SAFE.
        */
    void SetScaleFactor(const int NewScaleFactor) { ScaleFactor = NewScaleFactor; }

    /** \brief SubVec:  Computes a - b and puts it into c.
        */
    void SubVec(int a[3], int b[3], int c[3]);

    /** \brief Volumei returns the volume of the tetrahedron determined by f
        and p.
        */
    int Volumei(tFace f, tVertex p);

    /** \brief VolumeSign returns the sign of the volume of the tetrahedron determined by f
        and p.  VolumeSign is +1 iff p is on the negative side of f,
        where the positive side is determined by the rh-rule.  So the volume
        is positive if the ccw normal to f points outside the tetrahedron.
        The final fewer-multiplications form is due to Bob Williamson.
        */
    int VolumeSign(tFace f, tVertex p);

  private:
    bool debug;
    bool check;

    int ScaleFactor; // Scale factor to be used when converting from floating point to integers and vice versa
};

} // namespace AlignmentSubsystem
} // namespace INDI