Цитата: Если файл us_array.c имеет к этому отношениеб и, если пренебрежение авторскими правами Zemax Corp. Вас не сильно шокирует, не могли бы Вы бросить его мне в почту
us_array.c - лежит в папке установленного Zemax в качестве примера и реализует прямоугольный линзовый растр. Найти его можно в любом дистрибутиве Zemax. Если вы не пользуетесь Zemax, вам он мало поможет т.к. представляет собой исходник dll для Zemax.
Текст файла ниже:
Код: #include <windows.h>
#include <math.h>
#include <string.h>
#include "usersurf.h"
/*
Written by Kenneth E. Moore
Oct 11, 1996
This DLL models an arbitrary number of lens elements in a lens array.
The individual lenses are generally conic aspheres.
The user provides the number of elements in x and y, the size in x and y of
each element, and the radius, conic, and glass.
This surface breaks up the beam into numerous separate
beams, and so most ZEMAX features will fail to work with this surface.
However, the spot diagrams, image analysis, etc, all work okay.
Modified GetCellCenter 9-25-01 KEM to accept rays at edge of lenses
Modified GetCellCenter 2-07-06 KEM to bound cell values to valid range, useful for very fast lenslets
*/
int __declspec(dllexport) APIENTRY UserDefinedSurface(USER_DATA *UD, FIXED_DATA *FD);
int GetCellCenter(int nx, int ny, double wx, double wy, double x, double y, double *cx, double *cy);
/* a generic Snells law refraction routine */
int Refract(double thisn, double nextn, double *l, double *m, double *n, double ln, double mn, double nn);
BOOL WINAPI DllMain (HANDLE hInst, ULONG ul_reason_for_call, LPVOID lpReserved)
{
return TRUE;
}
/* this DLL models a lens array surface type */
int __declspec(dllexport) APIENTRY UserDefinedSurface(USER_DATA *UD, FIXED_DATA *FD)
{
int i, nx, ny, error, miss_flag;
double p2, alpha, power, a, b, c, rad, casp, t, zc;
double wx, wy, cx, cy, x, y, z;
switch(FD->type)
{
case 0:
/* ZEMAX is requesting general information about the surface */
switch(FD->numb)
{
case 0:
/* ZEMAX wants to know the name of the surface */
/* do not exceed 12 characters */
strcpy(UD->string,"Lens Array");
break;
case 1:
/* ZEMAX wants to know if this surface is rotationally symmetric */
/* it is not, so return a null string */
UD->string[0] = '\0';
break;
case 2:
/* ZEMAX wants to know if this surface is a gradient index media */
/* it is not, so return a null string */
UD->string[0] = '\0';
break;
}
break;
case 1:
/* ZEMAX is requesting the names of the parameter columns */
/* the value FD->numb will indicate which value ZEMAX wants. */
/* they are all "Unused" for this surface type */
/* returning a null string indicates that the parameter is unused. */
switch(FD->numb)
{
case 1:
strcpy(UD->string, "Number X");
break;
case 2:
strcpy(UD->string, "Number Y");
break;
case 3:
strcpy(UD->string, "Width X");
break;
case 4:
strcpy(UD->string, "Height Y");
break;
default:
UD->string[0] = '\0';
break;
}
break;
case 2:
/* ZEMAX is requesting the names of the extra data columns */
/* the value FD->numb will indicate which value ZEMAX wants. */
/* they are all "Unused" for this surface type */
/* returning a null string indicates that the extradata value is unused. */
switch(FD->numb)
{
default:
UD->string[0] = '\0';
break;
}
break;
case 3:
/* ZEMAX wants to know the sag of the surface */
/* if there is an alternate sag, return it as well */
/* otherwise, set the alternate sag identical to the sag */
/* The sag is sag1, alternate is sag2. */
UD->sag1 = 0.0;
UD->sag2 = 0.0;
/* if a plane, just return */
if (FD->cv == 0) return(0);
/* figure out the center coordinates of which "cell" we are in */
nx = (int)FD->param[1];
ny = (int)FD->param[2];
wx = FD->param[3];
wy = FD->param[4];
x = UD->x;
y = UD->y;
/* make sure nx and ny are both odd, otherwise the chief ray is a problem... */
if (!nx&1)nx++;
if (!ny&1)ny++;
if (wx <= 0.0 || wy <= 0.0) return(-1);
error = GetCellCenter(nx, ny, wx, wy, UD->x, UD->y, &cx, &cy);
if (error) return(0);
/* offset the coordinates */
x -= cx;
y -= cy;
p2 = x*x + y*y;
alpha = 1 - (1+FD->k)*FD->cv*FD->cv*p2;
if (alpha < 0)
{
/* lens is not big enough to fill the cell, radius is too short */
/* assume it is plane between lenses */
alpha = 0.0;
}
UD->sag1 = (FD->cv*p2)/(1 + sqrt(alpha));
/* forget supporting a hyper hemisphere! */
UD->sag2 = UD->sag1;
break;
case 4:
/* ZEMAX wants a paraxial ray trace to this surface */
/* x, y, z, and the optical path are unaffected, at least for this surface type */
/* for paraxial ray tracing, the return z coordinate should always be zero. */
/* paraxial surfaces are always planes with the following normals */
/* for a lens array, only consider the single lens on axis */
UD->ln = 0.0;
UD->mn = 0.0;
UD->nn = -1.0;
power = (FD->n2 - FD->n1)*FD->cv;
if ((UD->n) != 0.0)
{
(UD->l) = (UD->l)/(UD->n);
(UD->m) = (UD->m)/(UD->n);
(UD->l) = (FD->n1*(UD->l) - (UD->x)*power)/(FD->n2);
(UD->m) = (FD->n1*(UD->m) - (UD->y)*power)/(FD->n2);
/* normalize */
(UD->n) = sqrt(1/(1 + (UD->l)*(UD->l) + (UD->m)*(UD->m) ) );
/* de-paraxialize */
(UD->l) = (UD->l)*(UD->n);
(UD->m) = (UD->m)*(UD->n);
}
break;
case 5:
/* ZEMAX wants a real ray trace to this surface */
/* clear the multiple intercept test flag */
miss_flag = 0;
if (FD->cv == 0.0)
{
outofbounds:;
UD->ln = 0.0;
UD->mn = 0.0;
UD->nn = -1.0;
if (Refract(FD->n1, FD->n2, &UD->l, &UD->m, &UD->n, UD->ln, UD->mn, UD->nn)) return(-FD->surf);
return(0);
}
/* okay, not a plane. */
nx = (int)FD->param[1];
ny = (int)FD->param[2];
wx = FD->param[3];
wy = FD->param[4];
x = UD->x;
y = UD->y;
/* make sure nx and ny are both odd, otherwise the chief ray is a problem... */
if (!nx&1)nx++;
if (!ny&1)ny++;
if (wx <= 0.0 || wy <= 0.0) return(-1);
error = GetCellCenter(nx, ny, wx, wy, UD->x, UD->y, &cx, &cy);
if (error) goto outofbounds;
try_again:;
/* offset the coordinates */
x -= cx;
y -= cy;
z = 0.0;
a = (UD->n) * (UD->n) * FD->k + 1;
b = ((UD->n)/FD->cv) - (x) * (UD->l) - (y) * (UD->m);
c = (x) * (x) + (y) * (y);
rad = b * b - a * c;
if (rad < 0) return(FD->surf); /* ray missed this surface */
if (FD->cv > 0) t = c / (b + sqrt(rad));
else t = c / (b - sqrt(rad));
(x) = (UD->l) * t + (x);
(y) = (UD->m) * t + (y);
(z) = (UD->n) * t + (z);
UD->path = t;
/* okay, if the ray makes a steep angle of intercept,
we may actually have hit the wrong element. Check it again! */
if (miss_flag == 0)
{
double new_cx, new_cy;
/* avoid infinite loop */
miss_flag = 1;
/* restore global coordinates prior to test */
UD->x = x + cx;
UD->y = y + cy;
error = GetCellCenter(nx, ny, wx, wy, UD->x, UD->y, &new_cx, &new_cy);
if (error) goto outofbounds;
if (new_cx != cx || new_cy != cy)
{
/* we hit the wrong one! */
/* go back to the tangent plane */
t = -t;
(x) = (UD->l) * t + (x);
(y) = (UD->m) * t + (y);
x = x + cx;
y = y + cy;
cx = new_cx;
cy = new_cy;
goto try_again;
}
}
zc = (z) * FD->cv;
rad = zc * FD->k * (zc * (FD->k + 1) - 2) + 1;
casp = FD->cv / sqrt(rad);
UD->ln = (x) * casp;
UD->mn = (y) * casp;
UD->nn = ((z) - ((1/FD->cv) - (z) * FD->k)) * casp;
/* restore coordinates */
UD->x = x + cx;
UD->y = y + cy;
UD->z = z;
if (Refract(FD->n1, FD->n2, &UD->l, &UD->m, &UD->n, UD->ln, UD->mn, UD->nn)) return(-FD->surf);
break;
case 6:
/* ZEMAX wants the index, dn/dx, dn/dy, and dn/dz at the given x, y, z. */
/* This is only required for gradient index surfaces, so return dummy values */
UD->index = FD->n2;
UD->dndx = 0.0;
UD->dndy = 0.0;
UD->dndz = 0.0;
break;
case 7:
/* ZEMAX wants the "safe" data. */
/* this is used by ZEMAX to set the initial values for all parameters and extra data */
/* when the user first changes to this surface type. */
/* this is the only time the DLL should modify the data in the FIXED_DATA FD structure */
FD->param[1] = 5;
FD->param[2] = 5;
if (FD->cv == 0.0) FD->param[3] = 100.0;
else FD->param[3] = 0.5/FD->cv;
FD->param[4] = FD->param[3];
for (i = 5; i <= 8; i++) FD->param[i] = 0.0;
for (i = 1; i <= 200; i++) FD->xdata[i] = 0.0;
break;
}
return 0;
}
int Refract(double thisn, double nextn, double *l, double *m, double *n, double ln, double mn, double nn)
{
double nr, cosi, cosi2, rad, cosr, gamma;
if (thisn != nextn)
{
nr = thisn / nextn;
cosi = fabs((*l) * ln + (*m) * mn + (*n) * nn);
cosi2 = cosi * cosi;
if (cosi2 > 1) cosi2 = 1;
rad = 1 - ((1 - cosi2) * (nr * nr));
if (rad < 0) return(-1);
cosr = sqrt(rad);
gamma = nr * cosi - cosr;
(*l) = (nr * (*l)) + (gamma * ln);
(*m) = (nr * (*m)) + (gamma * mn);
(*n) = (nr * (*n)) + (gamma * nn);
}
return 0;
}
int GetCellCenter(int nx, int ny, double wx, double wy, double x, double y, double *cx, double *cy)
{
double hwx, hwy, tx, ty;
int tnx, tny;
int mx, my;
*cx = 0.0;
*cy = 0.0;
tnx = 0;
tny = 0;
hwx = 0.5*wx;
hwy = 0.5*wy;
mx = (nx-1)/2; // the maximum legal cell number (plus or minus)
my = (ny-1)/2;
/* do cx */
if (fabs(x) > hwx)
{
tx = x;
tnx = 0;
if (x > 0)
{
while(tx > hwx)
{
tnx++;
tx -= wx;
}
}
else
{
while(tx < -hwx)
{
tnx--;
tx += wx;
}
}
*cx = tnx*wx;
}
/* do cy */
if (fabs(y) > hwy)
{
ty = y;
tny = 0;
if (y > 0)
{
while(ty > hwy)
{
tny++;
ty -= wy;
}
}
else
{
while(ty < -hwy)
{
tny--;
ty += wy;
}
}
*cy = tny*wy;
}
// Modified 2-07-06 KEM to bound cell values to valid range, useful for very fast lenslets
// new test to bound cell values to valid range
while (tnx > mx) tnx--;
while (tnx <-mx) tnx++;
while (tny > my) tny--;
while (tny <-my) tny++;
*cx = tnx*wx;
*cy = tny*wy;
/* are there this many cells? */
if (fabs(tnx) > (nx-1)/2 || fabs(tny) > (ny-1)/2) return -1;
else return 0;
}