Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F98977611
o_chires.c
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Subscribers
None
File Metadata
Details
File Info
Storage
Attached
Created
Sat, Jan 18, 04:47
Size
57 KB
Mime Type
text/x-c
Expires
Mon, Jan 20, 04:47 (1 d, 20 h)
Engine
blob
Format
Raw Data
Handle
23648092
Attached To
R1448 Lenstool-HPC
o_chires.c
View Options
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<float.h>
#include<fonction.h>
#include<constant.h>
#include<dimension.h>
#include<structure.h>
#include "lt.h"
#ifdef _OPENMP
#include "omp.h"
#endif
#define CHIRES
static
double
chi2SglImage
(
struct
galaxie
*
pima
,
struct
point
*
ps
);
static
int
chi2_img
(
double
*
chi2
,
double
*
lh0
);
static
int
chi2_src
(
double
*
chi2
,
double
*
lh0
,
double
*
np_b0
);
static
void
chi2_wl
(
double
*
chi2
,
double
*
lh0
);
static
void
srcfitLinFit
(
double
*
chi2
,
double
*
lh0
)
__attribute__
((
noinline
));
/********************************************************/
/* fonction: o_chi */
/* auteur: jpk */
/********************************************************
* Return the total chi2 value
*
* Global variables used :
* - chip, chia, chix, chiy, chil, chis, chi_im, M, G, I, lens, shm, nshmap, arclet
* multi, cl, narclet, nwarn, optim_z, elix, SC, amplifi_mat, amplifi_matinv
* map_p, map_axx, map_ayy
* - in sig2posS() : amplifi
* - in sig2posSj() : amplifi
* - in sig2posS4j() : G, lens, lens_table
* - in distcosmo1() : C
* - in e_zeroamp() : G, lens, lens_table
* - in weight_baryc() : amplifi, G, lens, lens_table, multi, C
* - in fmin_ell() : elix, SC, G, lens, lens_table
* - in o_flux() : multi, G, lens, lens_table
* - in o_chi_flux() : I
* - in amplif() : I, amplifi, G, lens, C, lens_table
* - in amplif_mat() : I, multi, amplifi_mat, G, lens, lens_table, C
* - in amplif_matinv() : I, multi, amplifi_matinv, G, lens, lens_table, C
* - in dratio() : C
* - in e_unmag() : G, lens, lens_table
* - in chi_invim() : pi, ps, M, iline, radial, tangent, nrline, ntline,
* CL, lens, flagr, flagt, G, lens_table
* - in sp_set() : v_xx, v_yy
* - in o_dpl() : G, lens, lens_table
* - in o_mag_m() : G, lens, lens_table
* - in unlens_bc() : nwarn, distmin, G, lens, lens_table
*/
#ifdef CHIRES
FILE
*
OUT
;
void
o_chires
(
char
*
filename
)
#else
// Return 1 if error, 0 otherwise
int
o_chi_lhood0
(
double
*
chi_ext
,
double
*
lhood0_ext
,
double
*
np_b0
)
#endif
{
/* variables externes */
extern
struct
g_mode
M
;
const
extern
struct
g_grille
G
;
const
extern
struct
g_image
I
;
extern
struct
g_dyn
Dy
;
//TV
const
extern
struct
pot
lens
[
NLMAX
];
extern
struct
shear
shm
[];
extern
struct
cline
cl
[];
extern
double
chil
,
chis
,
chi_im
,
chi_vel
,
chi_mass
;
//TV
extern
int
nshmap
;
// extern double amplifi[NFMAX][NIMAX];
extern
double
**
map_p
,
**
map_axx
,
**
map_ayy
;
extern
int
nplo
,
**
imuo
;
extern
double
drima
,
**
imo
,
**
wo
,
**
ero
,
**
soo
;
extern
struct
pixlist
plo
[];
/* local variables */
struct
point
A
,
B
,
D
;
struct
ellipse
ampli
;
double
chi_mult
,
chisrcfit
,
chish
;
double
chi
,
lhood0
;
double
x
,
qp
,
tp
;
int
i
,
j
;
#ifdef CHIRES
OUT
=
fopen
(
filename
,
"w"
);
double
*
np_b0
=
NULL
;
#endif
/*variables initialisation*/
chi
=
0.
;
chish
=
0.
;
chi_im
=
0.
;
chis
=
0.
;
chi_mult
=
0.
;
chisrcfit
=
0.
;
chil
=
0.
;
lhood0
=
0.
;
chi_vel
=
0.
;
chi_mass
=
0.
;
/* spline mapping */
if
(
lens
[
0
].
type
==
10
)
{
sp_set
(
map_p
,
G
.
nx
,
G
.
ny
,
map_axx
,
map_ayy
);
/*
wr_pot("p.ipx",map_p);
wr_mass("m.ipx",map_axx,map_ayy);
*/
}
/* if cleanset is true in cleanlens*/
if
(
M
.
iclean
!=
0
)
{
if
(
M
.
iclean
==
2
)
{
// Add up barycenter position to source positions
extern
struct
g_source
S
;
extern
struct
galaxie
source
[
NFMAX
];
extern
struct
galaxie
multi
[
NFMAX
][
NIMAX
];
const
extern
struct
g_image
I
;
struct
point
Bs
[
NFMAX
];
// list of source barycenters
struct
point
Ps
[
NIMAX
];
char
str
[
IDSIZE
],
*
pch
;
// get image identifier
if
(
I
.
n_mult
>
0
)
{
for
(
i
=
0
;
i
<
S
.
ns
;
i
++
)
{
// check if source is attached to an image
strcpy
(
str
,
source
[
i
].
n
);
pch
=
strtok
(
str
,
"_"
);
pch
=
strtok
(
NULL
,
"_"
);
if
(
pch
==
NULL
)
break
;
// no attachment to an image
// look for the corresponding image family
j
=
0
;
while
(
indexCmp
(
pch
,
multi
[
j
][
0
].
n
)
&&
j
<
I
.
n_mult
)
j
++
;
// add barycenter position
if
(
j
<
I
.
n_mult
)
{
o_dpl
(
I
.
mult
[
j
],
multi
[
j
],
Ps
,
np_b0
);
Ps
[
I
.
mult
[
j
]]
=
Ps
[
0
];
Bs
[
i
]
=
bcentlist
(
Ps
,
I
.
mult
[
j
]);
source
[
i
].
C
.
x
+=
Bs
[
i
].
x
;
source
[
i
].
C
.
y
+=
Bs
[
i
].
y
;
}
}
}
// Compute image
extern
struct
g_pixel
imFrame
;
extern
struct
galaxie
source
[
NFMAX
];
const
extern
struct
g_observ
O
;
double
dx
=
imFrame
.
xmax
-
imFrame
.
xmin
;
double
dy
=
imFrame
.
ymax
-
imFrame
.
ymin
;
int
verbose_save
=
M
.
verbose
;
M
.
verbose
=
0
;
//2Eric: You assume that imFrame.pixelx == imFrame.pixely ?
if
(
fabs
((
imFrame
.
pixelx
-
imFrame
.
pixely
)
/
imFrame
.
pixelx
)
>
1e-4
)
{
fprintf
(
stderr
,
"Error | imFrame.pixelx(%f) != imFrame.pixely(%f) but we assume that they are equal
\n
"
,
imFrame
.
pixelx
,
imFrame
.
pixely
);
exit
(
EXIT_FAILURE
);
}
o_pixel
(
ero
,
imFrame
.
nx
,
imFrame
.
ny
,
imFrame
.
pixelx
,
imFrame
.
xmin
,
imFrame
.
xmax
,
imFrame
.
ymin
,
imFrame
.
ymax
,
source
,
map_axx
,
map_ayy
);
//wrf_fits_abs("simu.fits", ero, imFrame.nx, imFrame.ny, imFrame.xmin, imFrame.xmax, imFrame.ymin, imFrame.ymax, M.ref_ra, M.ref_dec);
if
(
O
.
setseeing
)
d_seeing_omp
(
ero
,
imFrame
.
nx
,
imFrame
.
ny
,
imFrame
.
pixelx
);
if
(
O
.
bruit
)
d_bruiter_omp
(
ero
,
imFrame
.
nx
,
imFrame
.
ny
);
M
.
verbose
=
verbose_save
;
if
(
wo
!=
NULL
)
{
for
(
i
=
0
;
i
<
imFrame
.
ny
;
i
++
)
for
(
j
=
0
;
j
<
imFrame
.
nx
;
j
++
)
{
ero
[
i
][
j
]
-=
imo
[
i
][
j
];
chi_im
+=
ero
[
i
][
j
]
*
ero
[
i
][
j
]
*
wo
[
i
][
j
];
lhood0
-=
log
(
2.
*
M_PI
*
wo
[
i
][
j
]
);
}
}
else
{
for
(
i
=
0
;
i
<
imFrame
.
ny
;
i
++
)
for
(
j
=
0
;
j
<
imFrame
.
nx
;
j
++
)
{
ero
[
i
][
j
]
-=
imo
[
i
][
j
];
chi_im
+=
ero
[
i
][
j
]
*
ero
[
i
][
j
]
/
fabs
(
imo
[
i
][
j
]
+
1.
);
lhood0
+=
log
(
2.
*
M_PI
*
fabs
(
imo
[
i
][
j
]
+
1.
)
);
}
}
// restore relative source positions
if
(
I
.
n_mult
>
0
)
for
(
i
=
0
;
i
<
S
.
ns
;
i
++
)
{
// check if source is attached to an image
strcpy
(
str
,
source
[
i
].
n
);
pch
=
strtok
(
str
,
"_"
);
pch
=
strtok
(
NULL
,
"_"
);
if
(
pch
==
NULL
)
break
;
// no attachment to an image
// look for the corresponding image family
j
=
0
;
while
(
indexCmp
(
pch
,
multi
[
j
][
0
].
n
)
&&
j
<
I
.
n_mult
)
j
++
;
// add barycenter position
if
(
j
<
I
.
n_mult
)
{
source
[
i
].
C
.
x
-=
Bs
[
i
].
x
;
source
[
i
].
C
.
y
-=
Bs
[
i
].
y
;
}
}
}
else
/* chi_im is the error associated to the transformation image -> source*/
chi_im
=
chi_invim
(
imo
,
plo
,
nplo
,
drima
,
soo
,
ero
,
imuo
);
#ifdef CHIRES
fprintf
(
OUT
,
"chi_im %.2lf
\n
"
,
chi_im
);
#endif
}
/* ----------------------------------------------------
* Weak-Lensing chi2
* if arcletstat is true in image section*/
if
(
I
.
stat
!=
0
)
{
chi2_wl
(
&
chis
,
&
lhood0
);
#ifndef CHIRES
// check for NaN. It can happen in some very particular conditions
// - when for 1 arclet, grad2.a = -5.2e-08 (EJ 16/03/2011)
if
(
chis
!=
chis
)
return
(
1
);
#endif
}
/* if shearmap is true in image section*/
if
(
I
.
shmap
!=
0
)
{
chish
=
0
;
for
(
i
=
0
;
i
<
nshmap
;
i
++
)
{
ampli
=
e_unmag
(
&
shm
[
i
].
C
,
I
.
dl0ssh
,
I
.
dossh
,
I
.
zsh
);
qp
=
fabs
(
ampli
.
b
/
ampli
.
a
);
tp
=
fabs
(
1.
-
qp
*
qp
)
/
2.
/
qp
;
ampli
.
theta
+=
M_PI
/
2.
;
x
=
(
shm
[
i
].
mx
-
tp
*
cos
(
2.
*
ampli
.
theta
))
/
shm
[
i
].
err
;
chish
+=
x
*
x
;
x
=
(
shm
[
i
].
my
-
tp
*
sin
(
2.
*
ampli
.
theta
))
/
shm
[
i
].
err
;
chish
+=
x
*
x
;
}
chish
/=
nshmap
;
chis
+=
chish
;
}
/*end if (I.shmap!=0)*/
/* if I.n_mult is true in image keyword*/
if
(
I
.
n_mult
!=
0
)
{
if
(
I
.
forme
>=
0
)
// Source plan chi2
x
=
chi2_src
(
&
chi_mult
,
&
lhood0
,
np_b0
);
else
{
// Image plane chi2
tp
=
lhood0
;
x
=
chi2_img
(
&
chi_mult
,
&
lhood0
);
/*
//Uncomment if you want to chain image plane and source plane
if( x != 0. )
{
chi_mult = 0.; lhood0 = tp;
chi2_src(&chi_mult, &lhood0 );
}
*/
}
#ifndef CHIRES
// This case should never happens in o_chires().
// this #ifndef condition aims only at avoiding the compilation warning
if
(
x
!=
0
)
return
(
1
);
#endif
}
/*end of optimization with the arclets*/
// if there are points for source plane fitting
if
(
I
.
srcfit
!=
0
)
{
srcfitLinFit
(
&
chisrcfit
,
&
lhood0
);
#ifndef CHIRES
if
(
chisrcfit
==
-
1.
)
return
(
1
);
#endif
}
/* if there are constraints on the critical lines positions in image keyword*/
if
(
I
.
npcl
!=
0
)
{
#ifdef CHIRES
fprintf
(
OUT
,
"chi critical ligne
\n
"
);
#endif
double
ampa
,
ampb
,
d
;
chil
=
0.
;
for
(
i
=
0
;
i
<
I
.
npcl
;
i
++
)
{
A
.
x
=
cl
[
i
].
C
.
x
-
cl
[
i
].
dl
*
cos
(
cl
[
i
].
phi
);
A
.
y
=
cl
[
i
].
C
.
y
-
cl
[
i
].
dl
*
sin
(
cl
[
i
].
phi
);
B
.
x
=
cl
[
i
].
C
.
x
+
cl
[
i
].
dl
*
cos
(
cl
[
i
].
phi
);
B
.
y
=
cl
[
i
].
C
.
y
+
cl
[
i
].
dl
*
sin
(
cl
[
i
].
phi
);
ampa
=
e_amp
(
&
A
,
cl
[
i
].
dl0s
,
cl
[
i
].
dos
,
cl
[
i
].
z
);
ampb
=
e_amp
(
&
B
,
cl
[
i
].
dl0s
,
cl
[
i
].
dos
,
cl
[
i
].
z
);
if
(
ampa
*
ampb
<
0.
)
// look for the critical line position between A and B
// the precision if fixed by PREC_ZERO
D
=
e_zeroamp
(
A
,
B
,
cl
[
i
].
dl0s
,
cl
[
i
].
dos
,
cl
[
i
].
z
);
else
{
if
((
B
.
x
!=
A
.
x
)
&&
(
ampb
!=
ampa
))
D
.
x
=
(
ampb
*
A
.
x
-
ampa
*
B
.
x
)
/
(
ampb
-
ampa
);
else
D
.
x
=
A
.
x
;
if
((
B
.
y
!=
A
.
y
)
&&
(
ampb
!=
ampa
))
D
.
y
=
(
ampb
*
A
.
y
-
ampa
*
B
.
y
)
/
(
ampb
-
ampa
);
else
D
.
y
=
A
.
y
;
}
d
=
dist
(
D
,
cl
[
i
].
C
)
/
cl
[
i
].
dl
;
chil
+=
d
*
d
;
// add parity chi2 (side A must be +-, side B must be ++)
ampli
=
e_unmag
(
&
A
,
cl
[
i
].
dl0s
,
cl
[
i
].
dos
,
cl
[
i
].
z
);
if
(
ampli
.
a
<
0.
)
chil
+=
ampli
.
a
*
ampli
.
a
;
if
(
ampli
.
b
>
0.
)
chil
+=
ampli
.
b
*
ampli
.
b
;
ampli
=
e_unmag
(
&
B
,
cl
[
i
].
dl0s
,
cl
[
i
].
dos
,
cl
[
i
].
z
);
if
(
ampli
.
a
<
0.
)
chil
+=
ampli
.
a
*
ampli
.
a
;
if
(
ampli
.
b
<
0.
)
chil
+=
ampli
.
b
*
ampli
.
b
;
#ifdef CHIRES
fprintf
(
OUT
,
"%d %.3lf %.3lf %.2lf
\n
"
,
i
,
cl
[
i
].
C
.
x
,
cl
[
i
].
C
.
y
,
d
*
d
);
}
fprintf
(
OUT
,
"chil %.2lf
\n
"
,
chil
);
#else
}
#endif
}
///////////////////////////////////////
if
(
Dy
.
dynnumber
==
1
||
Dy
.
dynnumber
==
2
||
Dy
.
dynnumber
==
3
||
Dy
.
dynnumber
==
4
)
{
//extern struct pot lens[NLMAX];
double
vel_model
;
//TV Oct2011
double
mass_model
;
if
(
Dy
.
dynnumber
==
1
||
Dy
.
dynnumber
==
2
)
{
vel_model
=
lens
[
0
].
sigma
/
1.473972264
;
//Transformation of the velocity in "our" velocity
chi_vel
=
(
Dy
.
dynvel
-
vel_model
)
*
(
Dy
.
dynvel
-
vel_model
)
/
Dy
.
dynevel
/
Dy
.
dynevel
;
lhood0
+=
log
(
2.
*
M_PI
*
(
Dy
.
dynevel
*
Dy
.
dynevel
)
);
// NORMALIZATION
}
if
(
Dy
.
dynnumber
==
2
)
{
mass_model
=
mass2d_NFW
(
lens
[
0
].
sigma
,
Dy
.
refradius
,
lens
[
0
].
rckpc
);
//Analitical mass
chi_mass
=
(
Dy
.
indmass
-
mass_model
)
*
(
Dy
.
indmass
-
mass_model
)
/
Dy
.
indemass
/
Dy
.
indemass
;
lhood0
+=
log
(
2.
*
M_PI
*
(
Dy
.
indemass
*
Dy
.
indemass
)
);
// NORMALIZATION
}
if
(
Dy
.
dynnumber
==
3
)
{
mass_model
=
mass3d_NFW
(
lens
[
0
].
sigma
,
Dy
.
refradius
,
lens
[
0
].
rckpc
);
//Analitical mass
chi_mass
=
(
Dy
.
indmass
-
mass_model
)
*
(
Dy
.
indmass
-
mass_model
)
/
Dy
.
indemass
/
Dy
.
indemass
;
lhood0
+=
log
(
2.
*
M_PI
*
(
Dy
.
indemass
*
Dy
.
indemass
)
);
// NORMALIZATION
}
/* if (Dy.dynnumber == 4) //MAMPOST
{
Full MAMPOSSt will be implemented soon
}*/
#ifdef CHIRES
fprintf
(
OUT
,
"chi_vel %7.4lf
\n
"
,
chi_vel
);
fprintf
(
OUT
,
"chi_mass %7.4lf
\n
"
,
chi_mass
);
#endif
}
////////////////////////
chi
=
chi_im
+
chis
+
chisrcfit
+
chil
+
chi_mult
+
chi_vel
+
chi_mass
;
//TV //Min(chi_mult, 100000000.);
// printf("chi_mult %lf chisrcfit %lf chil %lf\n",chi_mult,chisrcfit,chil);
#ifdef CHIRES
fprintf
(
OUT
,
"
\n
chitot %.2lf
\n
"
,
chi
);
fprintf
(
OUT
,
"log(Likelihood) %.2lf
\n
"
,
-
0.5
*
(
chi
+
lhood0
));
fclose
(
OUT
);
#endif
/* NPRINTF(stderr,"chip=%.3lf",chip); */
/* NPRINTF(stderr,f"chia=%.3lf chip=%.3lf chix=%.3lf chiy=%.3lf\n",chia,chip,chix,chiy); */
#ifndef CHIRES
*
chi_ext
=
chi
;
*
lhood0_ext
=
lhood0
;
return
0
;
// OK
#endif
}
#ifndef CHIRES
// Return the chi2. Return -1 if error
double
o_chi
()
{
int
error
;
double
chi2
,
lhood0
;
error
=
o_chi_lhood0
(
&
chi2
,
&
lhood0
,
NULL
);
if
(
!
error
)
return
(
chi2
);
else
return
(
-
1
);
}
#ifdef __SEGER__O_CHI_TEST
static
int
o_lhood_count
=
0
;
struct
timeval
o_lhood_tv1
;
#endif
double
o_lhood
(
int
*
error
)
{
double
chi2
,
lhood0
;
*
error
=
o_chi_lhood0
(
&
chi2
,
&
lhood0
,
NULL
);
#ifdef __SEGER__O_CHI_TEST
o_lhood_count
++
;
fprintf
(
stdout
,
"%d %g %g
\n
"
,
o_lhood_count
,
chi2
,
lhood0
);
if
(
o_lhood_count
==
1
)
gettimeofday
(
&
o_lhood_tv1
,
NULL
);
if
(
o_lhood_count
==
200
)
{
struct
timeval
tv2
;
gettimeofday
(
&
tv2
,
NULL
);
fprintf
(
stdout
,
"time %g
\n
"
,
(
double
)(
tv2
.
tv_sec
-
o_lhood_tv1
.
tv_sec
)
+
1e-6
*
(
tv2
.
tv_usec
-
o_lhood_tv1
.
tv_usec
));
//just in case o_chires test
o_chires
(
"o_chires.txt"
);
exit
(
0
);
}
#endif
return
(
-
0.5
*
(
chi2
+
lhood0
));
}
#endif
/* Return the chi2 to a set of arclet according to the
* I.stat chosen method.
*/
static
void
chi2_wl
(
double
*
chi2
,
double
*
lh0
)
{
extern
struct
g_mode
M
;
const
extern
struct
g_image
I
;
extern
struct
g_source
S
;
extern
long
int
narclet
;
extern
struct
galaxie
arclet
[
NAMAX
];
long
int
i
;
double
ells
,
wht_ell
,
es1
,
es2
;
/*error and weight in ellipticity in the image plane*/
double
a
,
b
,
e
,
g
,
g1
,
g2
,
e1
,
e2
;
double
chi2i
;
double
x
,
qp
,
dp
,
tp
,
k
;
struct
ellipse
ell_sou
,
source
,
ampli
;
/* variables locales images */
struct
matrix
grad2
;
#ifdef CHIRES
double
ga
;
// shear,kappa,reduced shear estimator
fprintf
(
OUT
,
"chis arclets
\n
"
);
fprintf
(
OUT
,
" N ID z chi2 gamma 1-k g es1 es2
\n
"
);
#endif
if
(
I
.
stat
==
1
||
I
.
stat
==
2
)
{
for
(
i
=
0
;
i
<
narclet
;
i
++
)
{
/* optimisation du z */
if
(
I
.
stat
==
2
)
{
printf
(
"WARN: redshift optimisation with I.stat=2 not yet implemented.
\n
"
);
// if (indexCmp(I.nza, arclet[i].n)) //TO CHECK (previous version : I.nza!=i+1)
// {
// SC = arclet[i].C;
// ampli = e_unmag_gal(&arclet[i]);
// elix = fabs(arclet[i].eps *
// cos(2 * (arclet[i].E.theta - ampli.theta)));
// //arclet[i].dr =
// // zero_t(arclet[i].dr, arclet[i].dr + .05, fmin_ell);
// }
// else
// // all arclets are optimised but arclet[i]
// arclet[i].dr = I.drarclet;
}
/* calcul de ts optimise */
ampli
=
e_unmag_gal
(
&
arclet
[
i
]);
isoima
(
&
arclet
[
i
].
E
,
&
ampli
,
&
ell_sou
);
x
=
ell_sou
.
b
/
ell_sou
.
a
;
x
=
.5
*
(
1.
/
x
-
x
);
if
(
I
.
stat
==
2
)
{
// if (indexCmp(I.nza, arclet[i+1].n)) //TO CHECK (previous version : I.nza!=i+1)
// *chi2 += x * x;
// else
// {
// // less weight to strong lensing arcs
// *chi2 += 50.*x * x;
// }
}
else
*
chi2
+=
x
*
x
;
};
/*end for (i=0;i<narclet;i++)*/
}
/* if ((I.stat==1)||(I.stat==2))*/
else
if
(
I
.
stat
==
3
)
/* optimisation de l'orientation */
{
for
(
i
=
0
;
i
<
narclet
;
i
++
)
{
ampli
=
e_unmag_gal
(
&
arclet
[
i
]);
x
=
arclet
[
i
].
tau
*
sin
(
2.
*
(
arclet
[
i
].
E
.
theta
-
ampli
.
theta
));
*
chi2
+=
x
*
x
;
#ifdef CHIRES
fprintf
(
OUT
,
"%ld %s %.2lf
\n
"
,
i
,
arclet
[
i
].
n
,
x
*
x
);
#endif
}
}
else
if
(
I
.
stat
==
4
)
/* optimisation orientation + ellipticite*/
{
for
(
i
=
0
;
i
<
narclet
;
i
++
)
{
ampli
=
e_unmag_gal
(
&
arclet
[
i
]);
qp
=
fabs
(
ampli
.
b
/
ampli
.
a
);
dp
=
(
1.
+
qp
*
qp
)
/
2.
/
qp
;
tp
=
fabs
(
1.
-
qp
*
qp
)
/
2.
/
qp
;
x
=
dp
*
arclet
[
i
].
tau
*
cos
(
2.
*
(
arclet
[
i
].
E
.
theta
-
ampli
.
theta
))
-
tp
*
arclet
[
i
].
dis
;
*
chi2
+=
x
*
x
;
}
//chis /=narclet;
}
else
if
(
I
.
stat
==
5
)
/* optimisation orientation + ellipticite*/
{
for
(
i
=
0
;
i
<
narclet
;
i
++
)
{
ampli
=
e_unmag_gal
(
&
arclet
[
i
]);
isoima
(
&
arclet
[
i
].
E
,
&
ampli
,
&
source
);
x
=
fabs
(
(
source
.
a
*
source
.
a
-
source
.
b
*
source
.
b
)
/
2.
/
source
.
a
*
source
.
b
);
*
chi2
+=
x
*
x
/
0.001
;
}
//chis /=narclet;
}
else
if
(
I
.
stat
==
6
)
/* optimisation ellipticite a la Marshall*/
{
double
lh0_add
=
0
;
double
chi2_add
=
0
;
int
num_threads
=
1
;
#ifdef _OPENMP
num_threads
=
omp_get_max_threads
();
#endif
if
(
num_threads
>
narclet
/
100
)
num_threads
=
narclet
/
100
;
#ifdef CHIRES
num_threads
=
1
;
#endif
#pragma omp parallel for if (num_threads > 1) schedule(guided, 100) \
private(ampli, source, ells, es1, es2, chi2i) \
reduction(+: lh0_add, chi2_add) num_threads(num_threads)
for
(
i
=
0
;
i
<
narclet
;
i
++
)
{
// Get amplification matrix:
ampli
=
e_unmag_gal
(
&
arclet
[
i
]);
// WARNING! if you uncomment something here put
// "local variables" into private section of #pragma omp
// a = ampli.a; b = ampli.b;
// e = (a * a - b * b) / (a * a + b * b);
// g1 = e * cos(2. * ampli.theta);
// g2 = e * sin(2. * ampli.theta);
//
// a = arclet[i].E.a; b = arclet[i].E.b;
// e = (a * a - b * b) / (a * a + b * b);
// e1 = e * cos(2. * arclet[i].E.theta);
// e2 = e * sin(2. * arclet[i].E.theta);
//
// es1 = e1 - g1;
// es2 = e2 - g2;
// Apply amplification matrix to image shape to get source shape:
isoima
(
&
arclet
[
i
].
E
,
&
ampli
,
&
source
);
// In weak lensing regime, have that e = e_s + g, or that e_s = e - g
// i.e. in this limit source contains the residual that goes into chi-sq
ells
=
(
source
.
a
*
source
.
a
-
source
.
b
*
source
.
b
)
/
(
source
.
a
*
source
.
a
+
source
.
b
*
source
.
b
);
es1
=
ells
*
cos
(
2
*
source
.
theta
);
es2
=
ells
*
sin
(
2
*
source
.
theta
);
//WARNING! if you uncomment something here then put
//"local variables" into private section of #pragma omp
// Ellipticity compnents are the objects that are Gaussian-distributed:
// cos2th = cos(2.0*source.theta);
// sin2th = sin(2.0*source.theta);
// ells1 = ells*cos2th;
// ells2 = ells*sin2th;
// Product of Gaussian distributions centred on zero:
//
// WARNING! if you uncoment it then put chis to reduction statement
// chis += ells1*ells1*wht_ell + ells2*ells2*wht_ell;
chi2i
=
es1
*
es1
/
(
I
.
sig2ell
+
arclet
[
i
].
var1
);
chi2i
+=
es2
*
es2
/
(
I
.
sig2ell
+
arclet
[
i
].
var2
);
chi2_add
+=
chi2i
;
//if ( I.statmode == 1 || I.dsigell != -1 )
//{
lh0_add
+=
log
(
2
*
M_PI
*
(
I
.
sig2ell
+
arclet
[
i
].
var1
)
)
+
log
(
2
*
M_PI
*
(
I
.
sig2ell
+
arclet
[
i
].
var2
)
);
//}
// TODO: test this on simulated data...
#ifdef CHIRES
#pragma omp critical
{
ga
=
(
ampli
.
a
-
ampli
.
b
)
/
2.
;
// gamma_i
k
=
(
ampli
.
a
+
ampli
.
b
)
/
2.
;
// divided by 1-k_i
g
=
ga
/
k
;
NPRINTF
(
stdout
,
"INFO: compute chi2 for arclet %ld/%ld
\r
"
,
i
+
1
,
narclet
);
fprintf
(
OUT
,
"%6ld %6s %.3lf %6.4lf %6.3le %6.3le %6.3le %6.3le %6.3le
\n
"
,
i
,
arclet
[
i
].
n
,
arclet
[
i
].
z
,
chi2i
,
ga
,
k
,
g
,
es1
,
es2
);
}
#endif
}
// printf( "%lf\n", chi2_add);
*
chi2
+=
chi2_add
;
*
lh0
+=
lh0_add
;
}
#ifdef CHIRES
else
if
(
I
.
stat
==
7
||
I
.
stat
==
8
)
// optimisation ellipticite superior a la Marshall
#else
else
if
(
I
.
stat
==
7
)
// optimisation ellipticite superior a la Marshall
#endif
{
// TODO include shape estimation error in quadrature?
// wht_ell = 1. / I.sig2ell;
// Compute the normalisation factor
// *lh0 += narclet * log( 2 * M_PI * I.sig2ell );
double
lh0_add
=
0
;
double
chi2_add
=
0
;
int
num_threads
=
1
;
#ifdef _OPENMP
num_threads
=
omp_get_max_threads
();
#endif
if
(
num_threads
>
narclet
/
100
)
num_threads
=
narclet
/
100
;
#ifdef CHIRES
num_threads
=
1
;
#endif
#pragma omp parallel for if (num_threads > 1) schedule(guided, 100) \
private(grad2, k, g1, g2, g, e, e1, e2, x, es1, es2, chi2i) \
reduction(+: lh0_add, chi2_add) num_threads(num_threads)
for
(
i
=
0
;
i
<
narclet
;
i
++
)
{
grad2
=
e_grad2_gal
(
&
arclet
[
i
],
NULL
);
// Get amplification matrix:
grad2
.
a
/=
arclet
[
i
].
dos
;
grad2
.
b
/=
arclet
[
i
].
dos
;
grad2
.
c
/=
arclet
[
i
].
dos
;
// a = ampli.a; b = ampli.b;
// // k = 0.5 * (grad2.a + grad2.c) <--> k = 1 - 0.5 * (ampli.a + ampli.b)
// // gam = 0.5 * (ampli.a - ampli.c) <--> gam = sqrt(g1^2 + g2^2)
// // gam * cos(2*ampli.theta) <-> g1 = 0.5 * (grad2.c - grad2.a)
// // gam * sin(2*ampli.theta) <-> g2 = - grad2.b
k
=
0.5
*
(
grad2
.
a
+
grad2
.
c
);
g1
=
0.5
*
(
grad2
.
c
-
grad2
.
a
)
/
(
1.
-
k
);
// reduced shear
g2
=
-
grad2
.
b
/
(
1.
-
k
);
g
=
g1
*
g1
+
g2
*
g2
;
// e = (grad2.a - grad2.c) / (a + b); // g = gamma / (1-kappa)
// g1 = e * cos(2. * ampli.theta);
// g2 = e * sin(2. * ampli.theta);
// a = arclet[i].E.a; b = arclet[i].E.b;
// e = (a - b) / (a + b);
// e1 = e * cos(2. * arclet[i].E.theta);
// e2 = e * sin(2. * arclet[i].E.theta);
// a & b are gamma1 and gamma2 (calibrated ellipticities)
// In COSMOS : e = (a*a-b*b)/(a*a+b*b) (cos(2theta) + i sin(2theta))
// (See Bartelmann & Schneider 2001 : Eq. 4.6 pg 49)
e
=
(
arclet
[
i
].
E
.
a
*
arclet
[
i
].
E
.
a
-
arclet
[
i
].
E
.
b
*
arclet
[
i
].
E
.
b
)
/
(
arclet
[
i
].
E
.
a
*
arclet
[
i
].
E
.
a
+
arclet
[
i
].
E
.
b
*
arclet
[
i
].
E
.
b
);
e1
=
e
*
cos
(
2.
*
arclet
[
i
].
E
.
theta
);
e2
=
e
*
sin
(
2.
*
arclet
[
i
].
E
.
theta
);
g1
*=
-
1
;
g2
*=
-
1
;
/// to match Bartelmann definition in Eq 4.6
x
=
1.
+
g
-
2.
*
(
g1
*
e1
+
g2
*
e2
);
es1
=
(
e1
-
2.
*
g1
+
e1
*
(
g1
*
g1
-
g2
*
g2
)
+
2.
*
g1
*
g2
*
e2
)
/
x
;
es2
=
(
e2
-
2.
*
g2
-
e2
*
(
g1
*
g1
-
g2
*
g2
)
+
2.
*
g1
*
g2
*
e1
)
/
x
;
// Apply amplification matrix to image shape to get source shape:
//isoima(&arclet[i].E, &li, &source);
// In weak lensing regime, have that e = e_s + g, or that e_s = e - g
// i.e. in this limit source contains the residual that goes into chisq
//ells = (source.a - source.b) / (source.a + source.b);
//ells should be Rayleigh-distributed:
//*chi2 += ells * ells * wht_ell; // - 2.0 * log(ells);
chi2i
=
es1
*
es1
/
(
I
.
sig2ell
+
arclet
[
i
].
var1
);
chi2i
+=
es2
*
es2
/
(
I
.
sig2ell
+
arclet
[
i
].
var2
);
chi2_add
+=
chi2i
;
lh0_add
+=
log
(
2
*
M_PI
*
(
I
.
sig2ell
+
arclet
[
i
].
var1
)
)
+
log
(
2
*
M_PI
*
(
I
.
sig2ell
+
arclet
[
i
].
var2
)
);
#ifdef CHIRES
#pragma omp critical
{
ampli
=
e_unmag_gal
(
&
arclet
[
i
]);
ga
=
(
ampli
.
a
-
ampli
.
b
)
/
2.
;
// gamma_i
k
=
(
ampli
.
a
+
ampli
.
b
)
/
2.
;
// divided by 1-k_i
g
=
ga
/
k
;
NPRINTF
(
stdout
,
"INFO: compute chi2 for arclet %ld/%ld
\r
"
,
i
+
1
,
narclet
);
fprintf
(
OUT
,
"%6ld %6s %.3lf %6.4lf %6.3le %6.3le %6.3le %6.3le %6.3le
\n
"
,
i
,
arclet
[
i
].
n
,
arclet
[
i
].
z
,
chi2i
,
ga
,
k
,
g
,
es1
,
es2
);
}
#endif
}
*
chi2
+=
chi2_add
;
*
lh0
+=
lh0_add
;
}
// else
// {
// fprintf(stderr, "ERROR in o_chi/arclet: I.stat = %d\n", I.stat);
// exit(-1);
// }
#ifdef CHIRES
NPRINTF
(
stdout
,
"
\n
"
);
fprintf
(
OUT
,
"chis %.2lf
\n
"
,
*
chi2
);
fprintf
(
OUT
,
"log(Likelihood) %.2lf
\n
"
,
-
0.5
*
(
*
chi2
+
(
*
lh0
)));
#endif
}
/* Return the chi2 for a single image.
* Parameters :
* - pima : pointer to an observed single image
* - ps : pointer to the corresponding predicted source position
*/
static
double
chi2SglImage
(
struct
galaxie
*
pima
,
struct
point
*
ps
)
{
const
extern
struct
g_image
I
;
struct
bitriplet
Tsol
[
NIMAX
];
// list of triangles containing predicted
//arclets for a single obs image
struct
point
Bs
;
/*barycenter of a familly of I.mult[i] sources*/
double
chi2
,
I2x
,
I2y
,
dx
,
dy
;
int
j
,
nimages
;
#ifdef CHIRES
double
rmsi_tot
,
rmsi
,
chi22
;
rmsi_tot
=
0.
;
#endif
chi2
=
0
;
I2x
=
I2y
=
I
.
sig2pos
[
0
][
0
];
//just an over check
if
(
pima
->
gsource
!=
NULL
&&
pima
->
grid_dr
<
0
)
{
fprintf
(
stderr
,
"You should initializa galaxie::gsource as NULL
\n
"
);
exit
(
EXIT_FAILURE
);
}
//if it is first run we should initialize pima->gsource
if
(
pima
->
gsource
==
NULL
)
{
pima
->
gsource
=
(
struct
point
(
*
)[
NGGMAX
][
NGGMAX
])
malloc
(
sizeof
(
struct
point
[
NGGMAX
][
NGGMAX
]));
const
extern
struct
g_grille
G
;
// fprintf(stderr,"%d %d\n", NGGMAX, G.ngrid);
pima
->
grid_dr
=
-
1
;
}
// Check if the grid is initialize to the image redshift
// But we should reconstruct grid in any case
// if ( pima->grid_dr != pima->dr )
{
pima
->
grid_dr
=
pima
->
dr
;
e_unlensgrid
(
*
(
pima
->
gsource
),
pima
->
grid_dr
);
}
//raise chip to a high value if we have more than 1 image
nimages
=
inverse
(
(
const
struct
point
(
*
)[
NGGMAX
])
*
(
pima
->
gsource
),
ps
,
Tsol
);
if
(
nimages
>
1
)
{
if
(
fabs
(
I
.
forme
)
==
10
)
I2x
=
I2y
=
pima
->
E
.
a
*
pima
->
E
.
a
;
else
if
(
fabs
(
I
.
forme
)
==
11
)
{
I2x
=
pima
->
E
.
a
*
cos
(
pima
->
E
.
theta
);
I2y
=
pima
->
E
.
b
*
cos
(
pima
->
E
.
theta
);
I2x
=
I2x
*
I2x
;
I2y
=
I2y
*
I2y
;
}
// compute chip as the total distance between the predicted images
// and the observed image
for
(
j
=
0
;
j
<
nimages
;
j
++
)
{
Bs
=
barycentre
(
&
Tsol
[
j
].
i
);
dx
=
Bs
.
x
-
pima
->
C
.
x
;
dy
=
Bs
.
y
-
pima
->
C
.
y
;
chi2
+=
dx
*
dx
/
I2x
+
dy
*
dy
/
I2y
;
#ifdef CHIRES
chi22
=
dx
*
dx
/
I2x
+
dy
*
dy
/
I2y
;
rmsi
=
dx
*
dx
+
dy
*
dy
;
rmsi_tot
+=
rmsi
;
fprintf
(
OUT
,
" 0 %6s %.3lf %d %7.2lf %6.2lf %6.2lf %6.2lf %.2lf %.2lf %d
\n
"
,
pima
->
n
,
pima
->
z
,
1
,
chi22
,
dx
*
dx
/
I2x
,
dy
*
dy
/
I2y
,
0.
,
0.
,
sqrt
(
rmsi
),
0
);
#endif
}
}
#ifdef CHIRES
if
(
nimages
!=
0
)
rmsi_tot
=
sqrt
(
rmsi_tot
/
nimages
);
fprintf
(
OUT
,
" 0 %6s %.3lf %d %7.2lf %6.2lf %6.2lf %6.2lf %.2lf %.2lf %d
\n
"
,
pima
->
n
,
pima
->
z
,
nimages
,
chi2
,
0.
,
0.
,
0.
,
0.
,
rmsi_tot
,
0
);
#endif
return
(
chi2
);
}
/*******************************************************
* Optimization with the arclet shape in the SOURCE PLANE.
*******************************************************/
static
int
chi2_src
(
double
*
chi2
,
double
*
lh0
,
double
*
np_b0
)
{
const
extern
struct
g_grille
G
;
const
extern
struct
g_image
I
;
extern
double
chip
,
chia
,
chix
,
chiy
;
extern
struct
galaxie
multi
[
NFMAX
][
NIMAX
];
extern
struct
galaxie
source
[
NFMAX
];
const
extern
int
optim_z
;
extern
struct
matrix
amplifi_mat
[
NFMAX
][
NIMAX
],
amplifi_matinv
[
NIMAX
][
NIMAX
];
extern
struct
sigposStr
sigposAs
;
struct
point
Ps
[
NIMAX
];
struct
point
Bs
;
//barycenter of a familly of I.mult[i] sources
struct
galaxie
*
pima
;
// pointer to an arclet
double
MA
,
MB
,
MC
,
MD
;
double
sigx2
,
sigy2
,
da
,
fluxS
,
sig2flux
;
// ,siga2;
int
ntot
;
// total number of multiple images
double
wtot
;
// sum of images weight (multi[i][j].var1, default:1)
double
Ix
,
Iy
,
I2x
,
I2y
,
Ixy
;
//sigma and sigma^2 in the source plane
double
w2x
,
w2y
,
wxy
;
//1/sigma^2 in the image & source plane
double
lhood0
;
// local lh0
double
dx
,
dy
;
//error in position in the source plane
double
chipij
;
int
optEllip
;
// test accelerator (boolean)
#define NDXDY 300
double
tmp
[
NDXDY
];
int
dcnt
,
ndxdy
;
// image counter
register
int
i
,
j
,
k
;
#ifdef CHIRES
struct
point
multib
[
NIMAX
];
int
nimages
;
// number of valid images in multib
extern
int
nwarn
;
/*counts each time a barycenter is not found in a
small source triangle. see e_im_prec() function*/
char
fname
[
15
];
// root name of a multiple images system
double
chipi
,
chixi
,
chiyi
,
chiai
;
// per family chi2
double
chixij
,
chiyij
,
chiaij
;
// per image chi2
double
rmss
,
rmsi
;
// per family rms in source and image plane
double
rmssj
,
rmsij
;
// per image "rms" in source and image plane
double
rmss_tot
,
rmsi_tot
;
// image rms tot in source and image plane
int
nwarni
;
// +1 if error in predicted image
fprintf
(
OUT
,
"chi multiples
\n
"
);
fprintf
(
OUT
,
" N ID z Narcs chip chix chiy chia rmss rmsi dx dy nwarn
\n
"
);
rmss_tot
=
rmsi_tot
=
0
;
nwarn
=
0
;
#endif
//
double
d
[
40
];
// double sigposMean;
chip
=
chix
=
chiy
=
chia
=
0.
;
lhood0
=
0.
;
ntot
=
wtot
=
0.
;
// Just to prevent warning messages during compilation
// I2x and I2y are properly computed later
I2x
=
I2y
=
sigposAs
.
min
*
sigposAs
.
min
;
// Regularisation of sig2pos
// sigposMean=0.; ntot=0;
// for( i=0 ; i< I.n_mult; i++)
// for( j = 0; j < I.mult[i]; j++)
// {
// sigposMean+=sqrt(sig2pos[i][j]);
// ntot++;
// }
// sigposMean/=ntot;
// for( i=0 ; i< I.n_mult; i++)
// for( j = 0; j < I.mult[i]; j++)
// {
// dx=sqrt(sig2pos[i][j])-sigposMean;
// chip+=dx*dx/I2x;
// }
// *lh0 += log( 2.*M_PI*I2x );
//
// It's not very necessary for I.forme==5 but it can be worth that
// in the rest of the code all the images have their amplification
// computed at some point, even if it's not necessary today (EJ: 07/07).
// if ( I.forme <= 7 || I.forme == 10 )
// amplif(np_b0, amplifi); //amplification of each image of all families
// EJ(22-05-2011): amplifi is removed and a call to e_amp_gal is made in each function which requires it
if
(
I
.
forme
==
8
||
I
.
forme
==
12
||
I
.
forme
==
13
)
amplif_mat
();
if
(
I
.
forme
==
9
)
amplif_matinv
();
// compute the total number of images, and initialise tmp[]
if
(
I
.
forme
==
12
)
{
ndxdy
=
0
;
dcnt
=
0
;
for
(
i
=
0
;
i
<
I
.
n_mult
&&
ndxdy
<
NDXDY
;
i
++
)
for
(
j
=
0
;
j
<
I
.
mult
[
i
]
&&
ndxdy
<
NDXDY
;
j
++
)
{
tmp
[
ndxdy
]
=
tmp
[
ndxdy
+
1
]
=
0.
;
ndxdy
+=
2
;
}
if
(
ndxdy
>=
NDXDY
)
{
fprintf
(
stderr
,
"ERROR: Too many images. Increase NDXDY in o_chi.c
\n
"
);
exit
(
0
);
}
}
// Define the number of threads we want to use
int
num_threads
=
1
;
#ifdef _OPENMP
num_threads
=
omp_get_max_threads
();
#endif
#ifdef CHIRES
num_threads
=
1
;
#endif
if
(
I
.
forme
==
12
)
num_threads
=
1
;
/* for each familly of arclets*/
#pragma omp parallel for if (num_threads > 1) \
private(optEllip,fluxS,i,I2x,I2y,Ix,Iy,Ps,pima,Bs, \
j,chipij,dx,dy,MA,MB,MC,MD, \
w2x,w2y,wxy,k,tmp,d,sigx2,sigy2,da,sig2flux) \
reduction(+: lhood0,chip,chix,chiy,chia,wtot,ntot) num_threads(num_threads)
for
(
i
=
0
;
i
<
I
.
n_mult
;
i
++
)
{
#ifdef CHIRES
chipi
=
chixi
=
chiyi
=
chiai
=
0.
;
rmss
=
rmsi
=
0.
;
nwarni
=
0
;
#endif
//int n_famille = i;
// Initialise shortcut tests
// for optimization with the ellipticity of the arclets
optEllip
=
0
;
if
(
(
I
.
forme
==
1
||
I
.
forme
==
3
)
&&
!
(
multi
[
i
][
0
].
E
.
a
==
multi
[
i
][
0
].
E
.
b
&&
multi
[
i
][
0
].
E
.
theta
==
0.
)
)
{
optEllip
=
1
;
o_mag_m
(
I
.
mult
[
i
],
multi
[
i
]);
}
/*optimization with the flux of the arclets*/
if
(
I
.
forme
==
2
||
I
.
forme
==
13
)
o_flux
(
I
.
mult
[
i
],
&
fluxS
,
i
,
np_b0
);
if
(
I
.
forme
<=
3
||
I
.
forme
==
10
)
// etendue method. Source plane error averaged
// over all the images of the system.
I2x
=
I2y
=
sig2posS
(
I
.
sig2pos
[
i
][
0
],
I
.
mult
[
i
],
i
,
np_b0
);
if
(
I
.
forme
==
10
)
Ix
=
I2x
/
I
.
sig2pos
[
i
][
0
];
// Ix: normalised etendue source plane
/*optim_z can be not null only with the o_runz*() functions*/
if
(
optim_z
==
0
)
/*Compute the sources positions in Ps */
o_dpl
(
I
.
mult
[
i
],
multi
[
i
],
Ps
,
np_b0
);
else
/*Compute the source position for each arclet*/
{
for
(
j
=
0
;
j
<
I
.
mult
[
i
];
j
++
)
{
pima
=
&
multi
[
i
][
j
];
Ps
[
j
].
x
=
pima
->
C
.
x
-
pima
->
Grad
.
x
*
multi
[
i
][
0
].
dr
;
Ps
[
j
].
y
=
pima
->
C
.
y
-
pima
->
Grad
.
y
*
multi
[
i
][
0
].
dr
;
}
Ps
[
I
.
mult
[
i
]]
=
Ps
[
0
];
/* NPRINTF(stderr," dr=%.3lf Ps=%.3lf\n",multi[i][0].dr,Ps[0].x); */
}
/*Find the barycenter position of the computed sources*/
if
(
I
.
forme
==
4
||
I
.
forme
==
6
)
Bs
=
weight_baryc
(
Ps
,
multi
[
i
],
I
.
mult
[
i
],
i
);
else
Bs
=
bcentlist
(
Ps
,
I
.
mult
[
i
]);
/* barycentre */
if
(
G
.
nlens
!=
G
.
nmsgrid
)
{
Bs
.
x
=
source
[
i
].
C
.
x
;
Bs
.
y
=
source
[
i
].
C
.
y
;
}
#ifdef CHIRES
/* Compute the rms in the image plane.
multib[] contains the predicted image positions
in source plane.
multib[] has the same size as multi[]
In case of lost images, the predicted image position is
the observed image position.
*/
int
det_stop
=
0
;
nimages
=
unlens_bc
(
Ps
,
Bs
,
multi
[
i
],
multib
,
I
.
mult
[
i
],
i
,
&
det_stop
);
#endif
// ****************************************************
// If we contraint with a SINGLE IMAGE...
// Exception!!! -->constraint in the image plane
//
if
(
I
.
mult
[
i
]
==
1
)
{
chip
+=
chi2SglImage
(
&
multi
[
i
][
0
],
&
Ps
[
0
]
)
*
multi
[
i
][
0
].
var1
;
wtot
+=
multi
[
i
][
0
].
var1
;
ntot
++
;
}
else
/******************************************************
* constraint with MULTIPLE IMAGES (SOURCE PLANE)
* in each familly i, for each arclet j, compute chip
**/
for
(
j
=
0
;
j
<
I
.
mult
[
i
];
j
++
)
{
// Initialisation of per image variables
chipij
=
0.
;
ntot
++
;
#ifdef CHIRES
chiaij
=
chixij
=
chiyij
=
0.
;
#endif
/* distance in arcsec in the source plane between barycenter
* and computed source position for each arclet*/
dx
=
Bs
.
x
-
Ps
[
j
].
x
;
dy
=
Bs
.
y
-
Ps
[
j
].
y
;
/* Modify the error in position according to the shear and
* convergence parameters in the source plane.
* (EJ 09/01/09: This is the proper way of computing the chi2,
* \theta_pred - \theta_obs = M^-1 ( \hat{\beta} - \beta_obs )
*/
if
(
I
.
forme
==
8
||
I
.
forme
==
13
)
{
double
cost
,
sint
,
a2
,
b2
;
double
dx_i
,
dy_i
;
double
wi2x
,
wi2y
,
wixy
;
MA
=
amplifi_mat
[
i
][
j
].
a
;
MB
=
amplifi_mat
[
i
][
j
].
b
;
MC
=
amplifi_mat
[
i
][
j
].
c
;
MD
=
amplifi_mat
[
i
][
j
].
d
;
dx_i
=
dx
*
MA
+
dy
*
MD
;
dy_i
=
dx
*
MB
+
dy
*
MC
;
// covariance matrix in amplification axis
cost
=
cos
(
multi
[
i
][
j
].
E
.
theta
);
sint
=
sin
(
multi
[
i
][
j
].
E
.
theta
);
// cost = 1.;
// sint = 0.;
a2
=
multi
[
i
][
j
].
E
.
a
*
multi
[
i
][
j
].
E
.
a
;
b2
=
multi
[
i
][
j
].
E
.
b
*
multi
[
i
][
j
].
E
.
b
;
wi2x
=
cost
*
cost
/
a2
+
sint
*
sint
/
b2
;
wixy
=
cost
*
sint
/
a2
-
cost
*
sint
/
b2
;
wi2y
=
cost
*
cost
/
b2
+
sint
*
sint
/
a2
;
// Not used in chi2 because of outflow errors in the denominator chi2
w2x
=
wi2x
*
MA
*
MA
+
2.
*
wixy
*
MA
*
MD
+
wi2y
*
MD
*
MD
;
w2y
=
wi2y
*
MC
*
MC
+
2.
*
wixy
*
MB
*
MC
+
wi2x
*
MB
*
MB
;
wxy
=
wi2x
*
MA
*
MB
+
wixy
*
(
MA
*
MC
+
MB
*
MD
)
+
wi2y
*
MC
*
MD
;
// chipij = dx*dx*wi2x*MA*MA + dx*dx*wi2y*MD*MD + 2.*dx*dx*wixy*MA*MD
// + dy*dy*wi2y*MC*MC + dy*dy*wi2x*MB*MB + 2.*dy*dy*wixy*MB*MC
// + 2.*(dx*dy*wi2x*MA*MB + dx*dy*wi2y*MC*MD + dx*dy*wixy*MA*MC + dx*dy*wixy*MB*MD);
// chipij = dx * dx * w2x + dy * dy * w2y + 2. * dx * dy * wxy;
chipij
=
dx_i
*
dx_i
*
wi2x
+
dy_i
*
dy_i
*
wi2y
+
2.
*
dx_i
*
dy_i
*
wixy
;
// update normalisation factor for image ij
// lhood0 += log( 4.*M_PI * M_PI / (w2x * w2y - wxy * wxy));
lhood0
+=
log
(
4.
*
M_PI
*
M_PI
/
(
wi2x
*
wi2y
-
wixy
*
wixy
));
// Detect infinite amplification issues
//if ( chipij < 0. ) return(1); // error
}
else
if
(
I
.
forme
==
12
)
// compute chi2 with covariance matrix
{
double
dx_i
,
dy_i
;
MA
=
amplifi_mat
[
i
][
j
].
a
;
MB
=
amplifi_mat
[
i
][
j
].
b
;
MC
=
amplifi_mat
[
i
][
j
].
c
;
MD
=
amplifi_mat
[
i
][
j
].
d
;
dx_i
=
dx
*
MA
+
dy
*
MD
;
dy_i
=
dx
*
MB
+
dy
*
MC
;
for
(
k
=
dcnt
+
1
;
k
<
ndxdy
;
k
++
)
tmp
[
k
]
+=
2.
*
dx_i
*
I
.
weight
[
dcnt
][
k
];
chipij
=
dx_i
*
dx_i
*
I
.
weight
[
dcnt
][
dcnt
]
+
tmp
[
dcnt
]
*
dx_i
;
d
[
dcnt
]
=
dx_i
;
dcnt
++
;
for
(
k
=
dcnt
+
1
;
k
<
ndxdy
;
k
++
)
tmp
[
k
]
+=
2.
*
dy_i
*
I
.
weight
[
dcnt
][
k
];
chipij
+=
dy_i
*
dy_i
*
I
.
weight
[
dcnt
][
dcnt
]
+
tmp
[
dcnt
]
*
dy_i
;
// lhood0 is updated once by detCov at the end
d
[
dcnt
]
=
dy_i
;
dcnt
++
;
}
else
if
(
G
.
nlens
!=
G
.
nmsgrid
)
{
Ix
=
multi
[
i
][
j
].
E
.
a
*
multi
[
i
][
j
].
mu
;
Iy
=
multi
[
i
][
j
].
E
.
b
*
multi
[
i
][
j
].
mu
;
I2x
=
Ix
*
Ix
;
I2y
=
Iy
*
Iy
;
chipij
=
dx
*
dx
/
I2x
+
dy
*
dy
/
I2y
;
lhood0
+=
log
(
4.
*
M_PI
*
M_PI
*
I2x
*
I2y
);
}
else
{
if
(
I
.
forme
==
4
||
I
.
forme
==
7
)
// sqrt(etendue) method. Image plane error = seeing
I2x
=
I2y
=
sig2posSj
(
I
.
sig2pos
[
i
][
j
],
multi
[
i
],
I
.
mult
[
i
],
j
,
i
);
else
if
(
I
.
forme
==
5
||
I
.
forme
==
6
)
// brute force method with 4 points
I2x
=
I2y
=
sig2posS4j
(
I
.
sig2pos
[
i
][
j
],
multi
[
i
],
Ps
[
j
],
j
);
else
if
(
I
.
forme
==
10
)
// etendue method. Different image plane errors
I2x
=
I2y
=
Ix
*
multi
[
i
][
j
].
E
.
a
*
multi
[
i
][
j
].
E
.
b
;
else
if
(
I
.
forme
==
11
)
// etendue method. Elliptical error in the source plane
// considering the image size as 1sigma error
sig2posSe
(
&
multi
[
i
][
j
],
&
I2x
,
&
I2y
);
w2x
=
1.
/
I2x
;
w2y
=
1.
/
I2y
;
wxy
=
0.
;
/*modify the sigma according to the shear and convergence
* parameters in the source plane*/
if
(
I
.
forme
==
9
)
{
MA
=
amplifi_matinv
[
i
][
j
].
a
;
MB
=
amplifi_matinv
[
i
][
j
].
b
;
MC
=
amplifi_matinv
[
i
][
j
].
c
;
MD
=
amplifi_matinv
[
i
][
j
].
d
;
Ix
=
(
MA
+
MD
)
*
sqrt
(
I
.
sig2pos
[
i
][
j
]);
Iy
=
(
MB
+
MC
)
*
sqrt
(
I
.
sig2pos
[
i
][
j
]);
I2x
=
Ix
*
Ix
;
I2y
=
Iy
*
Iy
;
w2x
=
1.
/
I2x
;
w2y
=
1.
/
I2y
;
}
chipij
=
dx
*
dx
*
w2x
+
dy
*
dy
*
w2y
+
2.
*
dx
*
dy
*
wxy
;
// update normalisation factor for image ij
lhood0
+=
log
(
4.
*
M_PI
*
M_PI
/
(
w2x
*
w2y
-
wxy
*
wxy
));
}
// sum total chip
chip
+=
chipij
*
multi
[
i
][
j
].
var1
;
wtot
+=
multi
[
i
][
j
].
var1
;
#ifdef CHIRES
// chi2 and rms for familly i
chipi
+=
chipij
;
rmssj
=
dx
*
dx
+
dy
*
dy
;
rmss
+=
rmssj
;
#endif
//optimization with the ellipticity of the arclets...
if
(
optEllip
)
{
o_shape
(
j
,
multi
[
i
],
&
dx
,
&
sigx2
,
&
dy
,
&
sigy2
,
&
da
);
chix
+=
dx
*
dx
/
sigx2
;
chiy
+=
dy
*
dy
/
sigy2
;
#ifdef CHIRES
chixij
=
dx
*
dx
/
sigx2
;
chixi
+=
chixij
;
chiyij
=
dy
*
dy
/
sigy2
;
chiyi
+=
chiyij
;
#endif
// ... and the flux
if
(
I
.
forme
==
3
)
{
chia
+=
da
*
da
/
I
.
sig2amp
;
#ifdef CHIRES
chiaij
=
da
*
da
/
I
.
sig2amp
;
chiai
+=
chiaij
;
#endif
}
}
//optimization with the flux of the arclets only
if
(
(
I
.
forme
==
2
||
I
.
forme
==
13
)
&&
multi
[
i
][
j
].
mag
!=
0.
)
{
o_chi_flux
(
&
multi
[
i
][
j
],
fluxS
,
&
da
,
&
sig2flux
);
chia
+=
da
*
da
/
sig2flux
;
lhood0
+=
log
(
2.
*
M_PI
*
sig2flux
);
#ifdef CHIRES
chiaij
=
da
*
da
/
sig2flux
;
chiai
+=
chiaij
;
#endif
}
#ifdef CHIRES
// compute the error in image plane from the multib[]
// computed earlier
dx
=
multib
[
j
].
x
-
multi
[
i
][
j
].
C
.
x
;
dy
=
multib
[
j
].
y
-
multi
[
i
][
j
].
C
.
y
;
rmsij
=
dx
*
dx
+
dy
*
dy
;
int
warnij
=
0
;
if
(
rmsij
>
0
)
rmsi
+=
rmsij
;
else
{
warnij
=
1
;
nwarni
+=
1
;
}
// Summarize all these calculations in one print
fprintf
(
OUT
,
"%2d %6s %.3lf 1 %7.2lf %6.2lf %6.2lf %6.2lf %6.3lf %6.2lf %6.2lf %6.2lf %d
\n
"
,
i
,
multi
[
i
][
j
].
n
,
multi
[
i
][
j
].
z
,
chipij
,
chixij
,
chiyij
,
chiaij
,
sqrt
(
rmssj
),
sqrt
(
rmsij
),
-
dx
,
dy
,
warnij
);
#endif
}
/*end for each image*/
#ifndef CHIRES
}
/*end for each familly i ( in o_chi() )*/
#else
// Finalise the statistics
if
(
nimages
-
nwarni
>
0
)
rmsi
/=
nimages
-
nwarni
;
else
rmsi
=
0
;
rmss
/=
I
.
mult
[
i
];
rmss_tot
+=
rmss
;
rmsi_tot
+=
rmsi
;
nwarn
+=
nwarni
;
strcpy
(
fname
,
multi
[
i
][
0
].
n
);
fname
[
strlen
(
multi
[
i
][
0
].
n
)
-
1
]
=
0
;
fprintf
(
OUT
,
"%2d %6s %.3lf %d %7.2lf %6.2lf %6.2lf %6.2lf %6.3lf %6.2lf N/A N/A %d
\n
"
,
i
,
fname
,
multi
[
i
][
0
].
z
,
I
.
mult
[
i
],
chipi
,
chixi
,
chiyi
,
chiai
,
sqrt
(
rmss
),
sqrt
(
rmsi
),
nwarni
);
}
/*end for each familly i ( in chires() ) */
#endif
// normalize chip, but we still want the full chi2, not the reduced chi2
chip
=
chip
*
ntot
/
wtot
;
#ifdef CHIRES
rmss_tot
/=
I
.
n_mult
;
rmsi_tot
/=
I
.
n_mult
;
fprintf
(
OUT
,
"chimul %7.2lf %6.2lf %6.2lf %6.2lf %6.3lf %6.2lf N/A N/A %d
\n
"
,
chip
,
chix
,
chiy
,
chia
,
sqrt
(
rmss_tot
),
sqrt
(
rmsi_tot
),
nwarn
);
fprintf
(
OUT
,
"log(Likelihood) %7.2lf
\n
"
,
-
0.5
*
(
chip
+
chia
+
chix
+
chiy
+
lhood0
));
#endif
// update the total statistics
if
(
I
.
forme
==
12
)
*
lh0
+=
ndxdy
*
log
(
2.
*
M_PI
)
+
log
(
I
.
detCov
);
else
*
lh0
+=
lhood0
;
*
chi2
=
chip
+
chia
+
chix
+
chiy
;
return
(
0
);
// OK
}
static
int
chi2_img
(
double
*
chi2
,
double
*
lh0
)
{
#ifndef CHIRES
extern
struct
g_mode
M
;
#endif
const
extern
struct
g_image
I
;
extern
double
chip
,
chia
,
chix
,
chiy
;
extern
struct
galaxie
multi
[
NFMAX
][
NIMAX
];
extern
int
nwarn
;
/*counts each time a barycenter is not found in a
small source triangle. see e_im_prec() function*/
const
extern
int
optim_z
;
#ifdef CHIRES
double
rmss_tot
,
rmsi_tot
;
// image rms tot in source and image plane
int
ntot
;
// total number of multiple images
fprintf
(
OUT
,
"chi multiples
\n
"
);
fprintf
(
OUT
,
" N ID z Narcs chip chix chiy chia rmss rmsi dx dy nwarn
\n
"
);
ntot
=
rmss_tot
=
rmsi_tot
=
0
;
#endif
chip
=
chia
=
chix
=
chiy
=
0.
;
nwarn
=
0
;
//I. We calculate numbers of all images in all families.
//It is equal to the number of times we should run unlens_bc_single
//or chi2SglImage
int
task_num
=
0
;
//number of time we should call unlens_bc_single
int
task_idx
;
//index of task
int
i
,
j
;
for
(
i
=
0
;
i
<
I
.
n_mult
;
i
++
)
task_num
+=
I
.
mult
[
i
];
struct
point
*
vPs
;
//array of source position for all images from all familes
struct
point
*
vmultib
;
//array of multib for all images from all families
struct
point
*
vBs
;
/*array of barycenters of a familly of I.mult[i] sources*/
int
*
task_imap
;
//task_imap[task_idx] = n_familly
int
*
task_jmap
;
//task_imap[task_idx] = j
vPs
=
(
struct
point
*
)
malloc
(
task_num
*
sizeof
(
struct
point
));
vmultib
=
(
struct
point
*
)
malloc
(
task_num
*
sizeof
(
struct
point
));
vBs
=
(
struct
point
*
)
malloc
(
I
.
n_mult
*
sizeof
(
struct
point
));
task_imap
=
(
int
*
)
malloc
(
task_num
*
sizeof
(
int
));
task_jmap
=
(
int
*
)
malloc
(
task_num
*
sizeof
(
int
));
#ifdef CHIRES
int
*
task_ok
=
(
int
*
)
malloc
(
task_num
*
sizeof
(
int
));
//find or not image
#endif
//II. We create task_imap and task_jmap to be able to make single for-loop
//over all images from all families
task_idx
=
0
;
for
(
i
=
0
;
i
<
I
.
n_mult
;
i
++
)
{
for
(
j
=
0
;
j
<
I
.
mult
[
i
]
;
j
++
,
task_idx
++
)
{
task_imap
[
task_idx
]
=
i
;
task_jmap
[
task_idx
]
=
j
;
}
}
//III. We calculate vPs[task_idx] and vBs[i]
task_idx
=
0
;
for
(
i
=
0
;
i
<
I
.
n_mult
;
i
++
)
{
struct
point
Ps
[
NIMAX
+
1
];
/* optim_z can be different from 0 with the o_runz*() functions*/
if
(
optim_z
==
0
)
/* compute the sources positions in Ps*/
o_dpl
(
I
.
mult
[
i
],
multi
[
i
],
Ps
,
NULL
);
else
{
for
(
j
=
0
;
j
<
I
.
mult
[
i
];
j
++
)
{
struct
galaxie
*
pima
=
&
multi
[
i
][
j
];
Ps
[
j
].
x
=
pima
->
C
.
x
-
pima
->
Grad
.
x
*
multi
[
i
][
0
].
dr
;
Ps
[
j
].
y
=
pima
->
C
.
y
-
pima
->
Grad
.
y
*
multi
[
i
][
0
].
dr
;
}
Ps
[
I
.
mult
[
i
]]
=
Ps
[
0
];
}
/*vBs[i] contains the barycenter of the I.mult[i] sources positions Ps*/
vBs
[
i
]
=
bcentlist
(
Ps
,
I
.
mult
[
i
]);
for
(
j
=
0
;
j
<
I
.
mult
[
i
];
j
++
,
task_idx
++
)
{
vPs
[
task_idx
]
=
Ps
[
j
];
}
}
//IV. We run unlens_bc_single for all images from all families in
//the single for cycle. For single image families we run chi2SglImage.
//This part is the most time consuming part so we can parallelise it.
int
det_stop
=
0
;
// if det_stop then we should return -1
// Define the number of threads we want to use
int
num_threads
=
1
;
#ifdef _OPENMP
num_threads
=
omp_get_max_threads
();
#endif
if
(
num_threads
>
task_num
/
2
)
num_threads
=
task_num
/
2
;
#ifdef CHIRES
num_threads
=
1
;
#endif
#pragma omp parallel for schedule(dynamic,1) private(i,j) num_threads(num_threads)
for
(
task_idx
=
0
;
task_idx
<
task_num
;
task_idx
++
)
{
if
(
det_stop
)
continue
;
i
=
task_imap
[
task_idx
];
//number of family
j
=
task_jmap
[
task_idx
];
//number of image in family i
if
(
I
.
mult
[
i
]
==
1
)
// SINGLE IMAGE system
{
double
chip_tmp
=
chi2SglImage
(
&
multi
[
i
][
0
],
&
vPs
[
task_idx
]
);
#pragma omp atomic
chip
+=
chip_tmp
;
}
else
//MULTIPLE IMAGES system
{
/*return in vmultib[task_idx] an image for current arclet */
int
ok
=
unlens_bc_single
(
vPs
[
task_idx
],
vBs
[
i
],
&
multi
[
i
][
j
],
&
vmultib
[
task_idx
],
i
);
//if !ok then we cannot find this image
#ifdef CHIRES
task_ok
[
task_idx
]
=
ok
;
#endif
#ifndef CHIRES
if
(
!
ok
&&
M
.
inverse
>
2
)
{
#pragma omp atomic
det_stop
++
;
//we "return -1"
//flush(det_stop) surprisingly reduce performance,
//so we don't do it
}
if
(
det_stop
)
continue
;
#endif
}
}
//V. Now we can caluculate chi2 using constructed vmultib
task_idx
=
0
;
for
(
i
=
0
;
(
i
<
I
.
n_mult
)
&&
(
!
det_stop
);
i
++
)
{
if
(
I
.
mult
[
i
]
==
1
)
//SINGLE IMAGE system
{
//we already add results to chip
task_idx
++
;
}
else
//MULTIPLE IMAGES systems
{
double
I2x
,
I2y
;
//sigma and sigma^2
double
dLhood0
;
// lh0 for a given familly of multiple images (I.forme=10 || sigposAs.bk !=0)
double
dx
,
dy
;
//error in position in the source plane
int
j
;
#ifdef CHIRES
int
nimages
=
0
;
double
chipi
,
chixi
,
chiyi
,
chiai
;
double
rmss
,
rmsi
;
// image rms in source and image plane
chipi
=
chixi
=
chiyi
=
chiai
=
0.
;
rmss
=
rmsi
=
0.
;
#endif
dLhood0
=
0.
;
// In all cases
I2x
=
I2y
=
I
.
sig2pos
[
i
][
0
];
// For each image in familly i
for
(
j
=
0
;
j
<
I
.
mult
[
i
];
j
++
,
task_idx
++
)
{
//current multib ---> vmultib[task_idx]
//current Bs ---> Bs[i]
//current task_ok---> task_ok[task_idx]
struct
galaxie
*
pima
=
&
multi
[
i
][
j
];
if
(
I
.
forme
==
-
1
)
{
I2x
=
I2y
=
I
.
sig2pos
[
i
][
j
];
}
else
if
(
I
.
forme
==
-
10
)
{
I2x
=
I2y
=
pima
->
E
.
a
*
pima
->
E
.
b
;
}
else
if
(
I
.
forme
==
-
11
)
{
I2x
=
pima
->
E
.
a
;
//* cos(pima->E.theta);
I2y
=
pima
->
E
.
b
;
//* cos(pima->E.theta);
I2x
=
I2x
*
I2x
;
I2y
=
I2y
*
I2y
;
}
// update normalisation factor for image ij
dLhood0
+=
log
(
4.
*
M_PI
*
M_PI
*
(
I2x
*
I2y
)
);
dx
=
vmultib
[
task_idx
].
x
-
multi
[
i
][
j
].
C
.
x
;
dy
=
vmultib
[
task_idx
].
y
-
multi
[
i
][
j
].
C
.
y
;
chip
+=
dx
*
dx
/
I2x
+
dy
*
dy
/
I2y
;
#ifdef CHIRES
if
(
task_ok
[
task_idx
])
nimages
++
;
double
rmsij
,
rmssj
,
chipij
;
chipi
+=
dx
*
dx
/
I2x
+
dy
*
dy
/
I2y
;
chipij
=
dx
*
dx
/
I2x
+
dy
*
dy
/
I2y
;
rmsi
+=
dx
*
dx
+
dy
*
dy
;
rmsij
=
dx
*
dx
+
dy
*
dy
;
// source plane rmss
double
dxs
,
dys
;
dxs
=
vBs
[
i
].
x
-
vPs
[
task_idx
].
x
;
dys
=
vBs
[
i
].
y
-
vPs
[
task_idx
].
y
;
rmss
+=
dxs
*
dxs
+
dys
*
dys
;
rmssj
=
dxs
*
dxs
+
dys
*
dys
;
//TODO: do we need warnj variable here?
int
warnj
=
0
;
fprintf
(
OUT
,
"%2d %6s %.3lf 1 %7.2lf %6.2lf %6.2lf %6.2lf %6.3lf %6.2lf %6.2lf %6.2lf %d
\n
"
,
i
,
multi
[
i
][
j
].
n
,
multi
[
i
][
j
].
z
,
chipij
,
0.
,
0.
,
0.
,
sqrt
(
rmssj
),
sqrt
(
rmsij
),
-
dx
,
dy
,
warnj
);
#endif
}
//end "for j" cycle
// update the total statistics
*
lh0
+=
dLhood0
;
// TODO: Implement chix, chiy and chia calculation in image plane
/*
for (j=0;j<I.mult[i];j++)
{
da=diff_mag(multi[i][j],multib[j]);
chia += da*da/I.sig2amp;
};*/
#ifdef CHIRES
rmss_tot
+=
rmss
;
rmsi_tot
+=
rmsi
;
ntot
+=
nimages
;
nwarn
+=
I
.
mult
[
i
]
-
nimages
;
if
(
nimages
!=
0
)
{
rmss
=
rmss
/
nimages
;
rmsi
=
rmsi
/
nimages
;
}
fprintf
(
OUT
,
"%2d %6s %.3lf %d %7.2lf %6.2lf %6.2lf %6.2lf %6.3lf %6.2lf N/A N/A %d
\n
"
,
i
,
multi
[
i
][
0
].
n
,
multi
[
i
][
0
].
z
,
I
.
mult
[
i
],
chipi
,
chixi
,
chiyi
,
chiai
,
sqrt
(
rmss
),
sqrt
(
rmsi
),
I
.
mult
[
i
]
-
nimages
);
#endif
}
// end of case with I.mult[i] > 1
}
/*end for each familly*/
free
(
vPs
);
free
(
vmultib
);
free
(
vBs
);
free
(
task_imap
);
free
(
task_jmap
);
#ifdef CHIRES
free
(
task_ok
);
// Total statistics
rmss_tot
=
sqrt
(
rmss_tot
/
ntot
);
rmsi_tot
=
sqrt
(
rmsi_tot
/
ntot
);
fprintf
(
OUT
,
"chimul %7.2lf %6.2lf %6.2lf %6.2lf %6.3lf %6.2lf N/A N/A %d
\n
"
,
chip
,
chix
,
chiy
,
chia
,
rmss_tot
,
rmsi_tot
,
nwarn
);
fprintf
(
OUT
,
"log(Likelihood) %7.2lf
\n
"
,
-
0.5
*
(
chip
+
(
*
lh0
)));
#endif
if
(
det_stop
)
return
-
1
;
*
chi2
=
chip
+
chia
+
chix
+
chiy
;
#ifdef DEBUG
printf
(
"All images found!
\n
"
);
#endif
return
0
;
// no error, no warning
}
/*****************************************************************
* Send the points in srcfit[] to source plane and fit a line
* to the source plane points.
*/
static
void
srcfitLinFit
(
double
*
chi2
,
double
*
lh0
)
{
const
extern
struct
g_image
I
;
extern
struct
galaxie
*
srcfit
;
struct
point
P
[
NSRCFIT
];
double
wt
[
NSRCFIT
],
t
;
double
ss
,
sx
,
sy
,
st2
,
a
,
b
,
tmp
;
int
i
;
// Send points to source plane and compute source plane weights
o_dpl
(
I
.
nsrcfit
,
srcfit
,
P
,
NULL
);
// Algorithm from IDL linfit.pro
ss
=
sx
=
sy
=
0.
;
for
(
i
=
0
;
i
<
I
.
nsrcfit
;
i
++
)
{
wt
[
i
]
=
srcfit
[
i
].
E
.
a
*
srcfit
[
i
].
E
.
b
;
//* fabs(e_amp_gal(&srcfit[i]));
wt
[
i
]
=
1.
/
wt
[
i
];
ss
+=
wt
[
i
]
*
wt
[
i
];
sx
+=
P
[
i
].
x
*
wt
[
i
]
*
wt
[
i
];
sy
+=
P
[
i
].
y
*
wt
[
i
]
*
wt
[
i
];
}
b
=
0.
;
st2
=
0.
;
for
(
i
=
0
;
i
<
I
.
nsrcfit
;
i
++
)
{
t
=
(
P
[
i
].
x
-
sx
/
ss
)
*
wt
[
i
];
b
+=
t
*
P
[
i
].
y
*
wt
[
i
];
st2
+=
t
*
t
;
}
b
/=
st2
;
a
=
(
sy
-
sx
*
b
)
/
ss
;
#ifdef CHIRES
fprintf
(
OUT
,
"
\n
source plane fitting y = %lf + %lf * x
\n
"
,
a
,
b
);
fprintf
(
OUT
,
" N ID wt chi2
\n
"
);
#endif
// Compute chi2
for
(
i
=
0
;
i
<
I
.
nsrcfit
;
i
++
)
{
tmp
=
(
P
[
i
].
y
-
b
*
P
[
i
].
x
-
a
)
*
wt
[
i
];
#ifdef CHIRES
fprintf
(
OUT
,
"%2d %4s %.3lf %.2lf
\n
"
,
i
,
srcfit
[
i
].
n
,
wt
[
i
],
tmp
*
tmp
);
// (1. + a * a));
#endif
*
chi2
+=
tmp
*
tmp
;
/// ( 1. + a * a);
*
lh0
+=
log
(
2.
*
M_PI
/
wt
[
i
]
/
wt
[
i
]
);
}
#ifdef CHIRES
fprintf
(
OUT
,
"chisrcfit %.2lf
\n
"
,
*
chi2
);
fprintf
(
OUT
,
"log(Likelihood) %7.2lf
\n
"
,
-
0.5
*
(
*
chi2
+
(
*
lh0
)));
#endif
}
Event Timeline
Log In to Comment