libmysofa-sys 0.2.1

Low level binding for libmysofa library
Documentation
/*
 * neighbor.c
 *
 *  Created on: 17.01.2017
 *      Author: hoene
 */

#include "mysofa.h"
#include "mysofa_export.h"
#include "tools.h"
#include <float.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

MYSOFA_EXPORT struct MYSOFA_NEIGHBORHOOD *
mysofa_neighborhood_init(struct MYSOFA_HRTF *hrtf,
                         struct MYSOFA_LOOKUP *lookup) {
  return mysofa_neighborhood_init_withstepdefine(
      hrtf, lookup, MYSOFA_DEFAULT_NEIGH_STEP_ANGLE,
      MYSOFA_DEFAULT_NEIGH_STEP_RADIUS);
}

MYSOFA_EXPORT struct MYSOFA_NEIGHBORHOOD *
mysofa_neighborhood_init_withstepdefine(struct MYSOFA_HRTF *hrtf,
                                        struct MYSOFA_LOOKUP *lookup,
                                        float angleStep, float radiusStep) {
  int i, index;
  float *origin, *test;
  float radius, radius2;
  float theta;
  float phi;

  // distance (degree) beyond which neighbor search is abandonned
  float maxNeighborSearchAngle = 45;

  struct MYSOFA_NEIGHBORHOOD *neighbor =
      malloc(sizeof(struct MYSOFA_NEIGHBORHOOD));
  if (!neighbor)
    return NULL;

  neighbor->elements = hrtf->M;
  neighbor->index = malloc(sizeof(int) * neighbor->elements * 6);
  if (!neighbor->index) {
    free(neighbor);
    return NULL;
  }
  for (i = 0; i < neighbor->elements * 6; i++)
    neighbor->index[i] = -1;

  origin = malloc(sizeof(float) * hrtf->C);
  test = malloc(sizeof(float) * hrtf->C);

  for (i = 0; i < hrtf->M; i++) {
    memcpy(origin, hrtf->SourcePosition.values + i * hrtf->C,
           sizeof(float) * hrtf->C);
    convertCartesianToSpherical(origin, hrtf->C);

    if ((lookup->phi_max - lookup->phi_min) > FLT_MIN) {
      phi = angleStep;
      do {
        test[0] = origin[0] + phi;
        test[1] = origin[1];
        test[2] = origin[2];
        convertSphericalToCartesian(test, 3);
        index = mysofa_lookup(lookup, test);
        if (index != i) {
          neighbor->index[i * 6 + 0] = index;
          break;
        }
        phi += angleStep;
      } while (phi <= maxNeighborSearchAngle);

      phi = -angleStep;
      do {
        test[0] = origin[0] + phi;
        test[1] = origin[1];
        test[2] = origin[2];
        convertSphericalToCartesian(test, 3);
        index = mysofa_lookup(lookup, test);
        if (index != i) {
          neighbor->index[i * 6 + 1] = index;
          break;
        }
        phi -= angleStep;
      } while (phi >= -maxNeighborSearchAngle);
    }

    if ((lookup->theta_max - lookup->theta_min) > FLT_MIN) {
      theta = angleStep;
      do {
        test[0] = origin[0];
        test[1] = origin[1] + theta;
        test[2] = origin[2];
        convertSphericalToCartesian(test, 3);
        index = mysofa_lookup(lookup, test);
        if (index != i) {
          neighbor->index[i * 6 + 2] = index;
          break;
        }
        theta += angleStep;
      } while (theta <= maxNeighborSearchAngle);

      theta = -angleStep;
      do {
        test[0] = origin[0];
        test[1] = origin[1] + theta;
        test[2] = origin[2];
        convertSphericalToCartesian(test, 3);
        index = mysofa_lookup(lookup, test);
        if (index != i) {
          neighbor->index[i * 6 + 3] = index;
          break;
        }
        theta -= angleStep;
      } while (theta >= -maxNeighborSearchAngle);
    }

    if ((lookup->radius_max - lookup->radius_min) > FLT_MIN) {
      radius = radiusStep;
      do {
        test[0] = origin[0];
        test[1] = origin[1];
        radius2 = test[2] = origin[2] + radius;
        convertSphericalToCartesian(test, 3);
        index = mysofa_lookup(lookup, test);
        if (index != i) {
          neighbor->index[i * 6 + 4] = index;
          break;
        }
        radius += radiusStep;
      } while (radius2 <= lookup->radius_max + radiusStep);

      radius = -radiusStep;
      do {
        test[0] = origin[0];
        test[1] = origin[1];
        radius2 = test[2] = origin[2] + radius;
        convertSphericalToCartesian(test, 3);
        index = mysofa_lookup(lookup, test);
        if (index != i) {
          neighbor->index[i * 6 + 5] = index;
          break;
        }
        radius -= radiusStep;
      } while (radius2 >= lookup->radius_min - radiusStep);
    }
  }
  free(test);
  free(origin);
  return neighbor;
}

MYSOFA_EXPORT int *mysofa_neighborhood(struct MYSOFA_NEIGHBORHOOD *neighborhood,
                                       int index) {
  if (index < 0 || index >= neighborhood->elements)
    return NULL;
  return neighborhood->index + index * 6;
}

MYSOFA_EXPORT void
mysofa_neighborhood_free(struct MYSOFA_NEIGHBORHOOD *neighborhood) {
  if (neighborhood) {
    free(neighborhood->index);
    free(neighborhood);
  }
}