lammps-sys 0.6.0

Generates bindings to LAMMPS' C interface (with optional builds from source)
Documentation
/*Copyright (c) 2016 PM Larsen

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/

//todo: normalize vertices

#include "ptm_neighbour_ordering.h"
#include <cmath>
#include <cstring>
#include <cassert>
#include <algorithm>
#include <set>
#include <vector>
#include "ptm_constants.h"
#include "ptm_voronoi_cell.h"


namespace ptm {

typedef struct
{
        double area;
        double dist;
        int index;
        int inner;
        int32_t number;
        double offset[3];
} sorthelper_t;

typedef struct
{
        size_t index;
        int32_t number;
        double area;
        double offset[3];

} solidnbr_t;

static bool sorthelper_compare(sorthelper_t const& a, sorthelper_t const& b)
{
        if (a.area > b.area)
                return true;

        if (a.area < b.area)
                return false;

        if (a.dist < b.dist)
                return true;

        return false;
}

static double dot_product(double* a, double* b)
{
        return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
}

static void cross_product(double* a, double* b, double* c)
{
        c[0] = a[1] * b[2] - a[2] * b[1];
        c[1] = a[2] * b[0] - a[0] * b[2];
        c[2] = a[0] * b[1] - a[1] * b[0];
}

static double calculate_solid_angle(double* R1, double* R2, double* R3)        //norms of R1-R3 must be 1
{
        double R2R3[3];
        cross_product(R2, R3, R2R3);
        double numerator = dot_product(R1, R2R3);

        double r1r2 = dot_product(R1, R2);
        double r2r3 = dot_product(R2, R3);
        double r3r1 = dot_product(R3, R1);

        double denominator = 1 + r1r2 + r3r1 + r2r3;
        return fabs(2 * atan2(numerator, denominator));
}

//todo: change voronoi code to return errors rather than exiting
static int calculate_voronoi_face_areas(int num_points, const double (*_points)[3], double* normsq, double max_norm, ptm_voro::voronoicell_neighbor* v, bool calc_solid_angles,
                                                std::vector<int>& nbr_indices, std::vector<double>& face_areas)
{
        const double k = 10 * max_norm;
        v->init(-k,k,-k,k,-k,k);

        for (int i=1;i<num_points;i++)
        {
                double x = _points[i][0] - _points[0][0];
                double y = _points[i][1] - _points[0][1];
                double z = _points[i][2] - _points[0][2];
                v->nplane(x,y,z,normsq[i],i);
        }

        v->neighbors(nbr_indices);

//v->face_areas(face_areas);
        if (!calc_solid_angles)
        {
                v->face_areas(face_areas);
                return 0;
        }
        else
        {
                std::vector<int> face_vertices;
                std::vector<double> vertices;

                v->face_vertices(face_vertices);
                v->vertices(0, 0, 0, vertices);

                size_t num_vertices = vertices.size() / 3;
                for (size_t i=0;i<num_vertices;i++)
                {
                        double x = vertices[i * 3 + 0];
                        double y = vertices[i * 3 + 1];
                        double z = vertices[i * 3 + 2];

                        double s = sqrt(x*x + y*y + z*z);
                        vertices[i * 3 + 0] /= s;
                        vertices[i * 3 + 1] /= s;
                        vertices[i * 3 + 2] /= s;
                }

                int num_faces = v->number_of_faces();

#ifdef DEBUG
                printf("number of voronoi faces: %d\n", num_faces);
#endif

//std::vector<double> solids(face_areas.size()+1);

                size_t c = 0;
                for (int current_face=0;current_face<num_faces;current_face++)
                {
                        int num = face_vertices[c++];

                        int point_index = nbr_indices[current_face];
                        if (point_index > 0)
                        {
                                double solid_angle = 0;
                                int u = face_vertices[c];
                                int v = face_vertices[c+1];
                                for (int i=2;i<num;i++)
                                {
                                        int w = face_vertices[c+i];
                                        double omega = calculate_solid_angle(&vertices[u*3], &vertices[v*3], &vertices[w*3]);
                                        solid_angle += omega;

                                        v = w;
                                }

                                face_areas[current_face] = solid_angle;
//solids[current_face] = solid_angle;
                                //face_areas[point_index] = solid_angle;
                        }

                        c += num;
                }

#ifdef DEBUG
printf("\n");
for (int i=0;i<solids.size();i++)
{
        printf("%d\t%f\t%f\n", i, solids[i], face_areas[i]);
}
#endif

                assert(c == face_vertices.size());
                return 0;
        }
}

static int _calculate_neighbour_ordering(void* _voronoi_handle, int num_points, double (*_points)[3], bool calc_solid_angles, sorthelper_t* data)
{
        assert(num_points <= PTM_MAX_INPUT_POINTS);

        ptm_voro::voronoicell_neighbor* voronoi_handle = (ptm_voro::voronoicell_neighbor*)_voronoi_handle;

        double max_norm = 0;
        double points[PTM_MAX_INPUT_POINTS][3];
        double normsq[PTM_MAX_INPUT_POINTS];

        for (int i=0;i<num_points;i++)
        {
                double x = _points[i][0] - _points[0][0];
                double y = _points[i][1] - _points[0][1];
                double z = _points[i][2] - _points[0][2];
                points[i][0] = x;
                points[i][1] = y;
                points[i][2] = z;

                normsq[i] = x*x + y*y + z*z;
                max_norm = std::max(max_norm, normsq[i]);
        }

        max_norm = sqrt(max_norm);

        std::vector<int> nbr_indices(num_points + 6);
        std::vector<double> face_areas(num_points + 6);
        int ret = calculate_voronoi_face_areas(num_points, points, normsq, max_norm, voronoi_handle, calc_solid_angles, nbr_indices, face_areas);
        if (ret != 0)
                return ret;

        double areas[PTM_MAX_INPUT_POINTS];
        memset(areas, 0, num_points * sizeof(double));
        areas[0] = INFINITY;
        for (size_t i=0;i<nbr_indices.size();i++)
        {
                int index = nbr_indices[i];
                if (index > 0)
                        areas[index] = face_areas[i];
        }

        for (int i=0;i<num_points;i++)
        {
                assert(areas[i] == areas[i]);
                data[i].area = areas[i];
                data[i].dist = normsq[i];
                data[i].index = i;
                memcpy(data[i].offset, points[i], 3 * sizeof(double));
        }

        std::sort(data, data + num_points, &sorthelper_compare);
        return ret;
}

int calculate_neighbour_ordering(        void* _voronoi_handle, size_t atom_index, int min_points, int (get_neighbours)(void* vdata, size_t central_index, size_t atom_index, int num, size_t* nbr_indices, int32_t* numbers,
                                        double (*nbr_pos)[3]), void* nbrlist, bool calc_solid_angles,
                                        size_t* indices, double (*points)[3], int32_t* numbers)
{
        size_t nbr_indices[PTM_MAX_INPUT_POINTS];
        int32_t nbr_numbers[PTM_MAX_INPUT_POINTS];
        double nbr_pos[PTM_MAX_INPUT_POINTS][3];
        int num_points = get_neighbours(nbrlist, atom_index, atom_index, PTM_MAX_INPUT_POINTS, nbr_indices, nbr_numbers, nbr_pos);
        if (num_points < min_points)
                return -1;

        sorthelper_t data[PTM_MAX_INPUT_POINTS];
        int ret = _calculate_neighbour_ordering(_voronoi_handle, num_points, nbr_pos, calc_solid_angles, data);
        if (ret != 0)
                return ret;

        for (int i=0;i<num_points;i++)
        {
                int index = data[i].index;
                indices[i] = nbr_indices[index];
                numbers[i] = nbr_numbers[index];
                memcpy(&points[i], nbr_pos[index], 3 * sizeof(double));
        }

        return num_points;
}

static int find_diamond_neighbours(void* _voronoi_handle, int num_points, double (*_points)[3], size_t* nbr_indices, int32_t* nbr_numbers, int num_solid_nbrs, bool calc_solid_angles, solidnbr_t* nbrlist)
{
        sorthelper_t data[PTM_MAX_INPUT_POINTS];
        int ret = _calculate_neighbour_ordering(_voronoi_handle, num_points, _points, calc_solid_angles, data);
        if (ret != 0)
                return ret;

        int n = std::min(num_solid_nbrs, num_points - 1);
        for (int i=0;i<n;i++)
        {
                nbrlist[i].index = nbr_indices[data[i+1].index];
                nbrlist[i].area = data[i+1].area;
                nbrlist[i].number = nbr_numbers[data[i+1].index];
                memcpy(nbrlist[i].offset, data[i+1].offset, 3 * sizeof(double));
        }

        return ret;
}

void* voronoi_initialize_local()
{
        ptm_voro::voronoicell_neighbor* ptr = new ptm_voro::voronoicell_neighbor;
        return (void*)ptr;
}

void voronoi_uninitialize_local(void* _ptr)
{
        ptm_voro::voronoicell_neighbor* ptr = (ptm_voro::voronoicell_neighbor*)_ptr;
        delete ptr;
}

#define MAX_INNER 4
#define MAX_SNBRS 12

int calculate_two_shell_neighbour_ordering(        void* _voronoi_handle, size_t atom_index, int (get_neighbours)(void* vdata, size_t central_index, size_t atom_index, int num, size_t* nbr_indices, int32_t* numbers, double (*nbr_pos)[3]), void* nbrlist,
                                                int num_inner, int num_outer, int max_snbrs, bool calc_solid_angles,
                                                size_t* nbr_indices, double (*points)[3], int32_t* numbers)
{
        assert(num_inner <= MAX_INNER);

        size_t central_nbr_indices[MAX_SNBRS + 1];
        int32_t central_nbr_numbers[MAX_SNBRS + 1];
        double central_nbr_pos[MAX_SNBRS + 1][3];
        int num_points = get_neighbours(nbrlist, atom_index, atom_index, max_snbrs + 1, central_nbr_indices, central_nbr_numbers, central_nbr_pos);
        if (num_points < num_inner + 1)
                return -1;

        solidnbr_t central_solid[MAX_SNBRS];
        int ret = find_diamond_neighbours(_voronoi_handle, num_points, central_nbr_pos, central_nbr_indices, central_nbr_numbers, max_snbrs, calc_solid_angles, central_solid);
        if (ret != 0)
                return ret;


        sorthelper_t data[MAX_INNER * 6];
        nbr_indices[0] = atom_index;
        numbers[0] = central_nbr_numbers[0];
        memset(&points[0], 0, 3 * sizeof(double));

        std::set<size_t> claimed;
        claimed.insert(atom_index);
        for (int i=0;i<num_inner;i++)
        {
                nbr_indices[i+1] = central_solid[i].index;
                numbers[i+1] = central_solid[i].number;
                memcpy(&points[i+1], central_solid[i].offset, 3 * sizeof(double));

                claimed.insert(central_solid[i].index);
        }

        int index = 0;
        for (int i=0;i<num_inner;i++)
        {
                size_t inner_index = central_solid[i].index;

                //----------------------------------------------
                size_t inner_nbr_indices[MAX_SNBRS + 1];
                int32_t inner_nbr_numbers[MAX_SNBRS + 1];
                double inner_nbr_pos[MAX_SNBRS + 1][3];
                int num_points = get_neighbours(nbrlist, atom_index, inner_index, max_snbrs + 1, inner_nbr_indices, inner_nbr_numbers, inner_nbr_pos);
                if (num_points < num_inner + 1)
                        return -1;

                solidnbr_t inner_solid[MAX_SNBRS];
                ret = find_diamond_neighbours(_voronoi_handle, num_points, inner_nbr_pos, inner_nbr_indices, inner_nbr_numbers, max_snbrs, calc_solid_angles, inner_solid);
                if (ret != 0)
                        return ret;
                //----------------------------------------------

                int n = std::min(6, num_points);
                for (int j=0;j<n;j++)
                {
                        bool already_claimed = claimed.find(inner_solid[j].index) != claimed.end();
                        if (already_claimed)
                                continue;

                        data[index].inner = i;
                        data[index].index = inner_solid[j].index;
                        data[index].area = inner_solid[j].area;
                        data[index].dist = 0;
                        data[index].number = inner_solid[j].number;

                        memcpy(data[index].offset, inner_solid[j].offset, 3 * sizeof(double));
                        for (int k=0;k<3;k++)
                                data[index].offset[k] += central_solid[i].offset[k];

                        index++;
                }
        }

        int n = index;
        std::sort(data, data + n, &sorthelper_compare);

        int num_found = 0;
        int counts[MAX_INNER] = {0};
        for (int i=0;i<n;i++)
        {
                int inner = data[i].inner;
                int nbr_atom_index = data[i].index;

                bool already_claimed = claimed.find(nbr_atom_index) != claimed.end();
                if (counts[inner] >= num_outer || already_claimed)
                        continue;

                nbr_indices[1 + num_inner + num_outer * inner + counts[inner]] = nbr_atom_index;
                numbers[1 + num_inner + num_outer * inner + counts[inner]] = data[i].number;
                memcpy(points[1 + num_inner + num_outer * inner + counts[inner]], &data[i].offset, 3 * sizeof(double));
                claimed.insert(nbr_atom_index);

                counts[inner]++;
                num_found++;
                if (num_found >= num_inner * num_outer)
                        break;
        }

        if (num_found != num_inner * num_outer)
                return -1;

        return 0;
}

}