Statistiques
| Révision :

root / TrouNoir / trou_noir.c @ 215

Historique | Voir | Annoter | Télécharger (9,95 ko)

1
/*
2
        Programme original realise en Fortran 77 en mars 1994
3
        pour les Travaux Pratiques de Modelisation Numerique
4
        DEA d'astrophysique et techniques spatiales de Meudon
5

6
                par Herve Aussel et Emmanuel Quemener
7

8
        Conversion en C par Emmanuel Quemener en aout 1997
9
        Modification par Emmanuel Quemener en aout 2019
10

11
        Remerciements a :
12

13
        - Herve Aussel pour sa procedure sur le spectre de corps noir
14
        - Didier Pelat pour l'aide lors de ce travail
15
        - Jean-Pierre Luminet pour son article de 1979
16
        - Le Numerical Recipies pour ses recettes de calcul
17
        - Luc Blanchet pour sa disponibilite lors de mes interrogations en RG
18

19
        Compilation sous gcc ( Compilateur GNU sous Linux ) :
20

21
                gcc -O2 -o trou_noir trou_noir.c -lm
22
*/ 
23

    
24
#include <stdio.h>
25
#include <math.h>
26
#include <stdlib.h>
27
#include <string.h>
28
#include <sys/time.h>
29
#include <time.h>
30

    
31
#define nbr 256 /* Nombre de colonnes du spectre */
32

    
33
#define PI 3.14159265359
34

    
35
#define TRACKPOINTS 2048
36

    
37
#if TYPE == FP32
38
#define MYFLOAT float
39
#else
40
#define MYFLOAT double
41
#endif
42

    
43
MYFLOAT atanp(MYFLOAT x,MYFLOAT y)
44
{
45
  MYFLOAT angle;
46

    
47
  angle=atan2(y,x);
48

    
49
  if (angle<0)
50
    {
51
      angle+=2*PI;
52
    }
53

    
54
  return angle;
55
}
56

    
57

    
58
MYFLOAT f(MYFLOAT v)
59
{
60
  return v;
61
}
62

    
63
MYFLOAT g(MYFLOAT u,MYFLOAT m,MYFLOAT b)
64
{
65
  return (3.*m/b*pow(u,2)-u);
66
}
67

    
68

    
69
void calcul(MYFLOAT *us,MYFLOAT *vs,MYFLOAT up,MYFLOAT vp,
70
            MYFLOAT h,MYFLOAT m,MYFLOAT b)
71
{
72
  MYFLOAT c[4],d[4];
73

    
74
  c[0]=h*f(vp);
75
  c[1]=h*f(vp+c[0]/2.);
76
  c[2]=h*f(vp+c[1]/2.);
77
  c[3]=h*f(vp+c[2]);
78
  d[0]=h*g(up,m,b);
79
  d[1]=h*g(up+d[0]/2.,m,b);
80
  d[2]=h*g(up+d[1]/2.,m,b);
81
  d[3]=h*g(up+d[2],m,b);
82

    
83
  *us=up+(c[0]+2.*c[1]+2.*c[2]+c[3])/6.;
84
  *vs=vp+(d[0]+2.*d[1]+2.*d[2]+d[3])/6.;
85
}
86

    
87
void rungekutta(MYFLOAT *ps,MYFLOAT *us,MYFLOAT *vs,
88
                MYFLOAT pp,MYFLOAT up,MYFLOAT vp,
89
                MYFLOAT h,MYFLOAT m,MYFLOAT b)
90
{
91
  calcul(us,vs,up,vp,h,m,b);
92
  *ps=pp+h;
93
}
94

    
95

    
96
MYFLOAT decalage_spectral(MYFLOAT r,MYFLOAT b,MYFLOAT phi,
97
                         MYFLOAT tho,MYFLOAT m)
98
{
99
  return (sqrt(1-3*m/r)/(1+sqrt(m/pow(r,3))*b*sin(tho)*sin(phi)));
100
}
101

    
102
MYFLOAT spectre(MYFLOAT rf,MYFLOAT q,MYFLOAT b,MYFLOAT db,
103
             MYFLOAT h,MYFLOAT r,MYFLOAT m,MYFLOAT bss)
104
{
105
  MYFLOAT flx;
106

    
107
  flx=exp(q*log(r/m))*pow(rf,4)*b*db*h;
108
  return(flx);
109
}
110

    
111
MYFLOAT spectre_cn(MYFLOAT rf,MYFLOAT b,MYFLOAT db,
112
                MYFLOAT h,MYFLOAT r,MYFLOAT m,MYFLOAT bss)
113
{
114
  
115
  MYFLOAT flx;
116
  MYFLOAT nu_rec,nu_em,qu,temp_em,flux_int;
117
  int fi,posfreq;
118

    
119
#define planck 6.62e-34
120
#define k 1.38e-23
121
#define c2 9.e16
122
#define temp 3.e7
123
#define m_point 1.
124

    
125
#define lplanck (log(6.62)-34.*log(10.))
126
#define lk (log(1.38)-23.*log(10.))
127
#define lc2 (log(9.)+16.*log(10.))
128
  
129
  MYFLOAT v=1.-3./r;
130

    
131
  qu=1./sqrt((1.-3./r)*r)*(sqrt(r)-sqrt(6.)+sqrt(3.)/2.*log((sqrt(r)+sqrt(3.))/(sqrt(r)-sqrt(3.))* 0.17157287525380988 ));
132

    
133
  temp_em=temp*sqrt(m)*exp(0.25*log(m_point)-0.75*log(r)-0.125*log(v)+0.25*log(fabs(qu)));
134

    
135
  flux_int=0.;
136
  flx=0.;
137

    
138
  for (fi=0;fi<nbr;fi++)
139
    {
140
      nu_em=bss*(MYFLOAT)fi/(MYFLOAT)nbr;
141
      nu_rec=nu_em*rf;
142
      posfreq=(int)(nu_rec*(MYFLOAT)nbr/bss);
143
      if ((posfreq>0)&&(posfreq<nbr))
144
          {
145
          flux_int=2.*planck/c2*pow(nu_em,3)/(exp(planck*nu_em/(k*temp_em))-1.)*exp(3.*log(rf))*b*db*h;
146
            flx+=flux_int;
147
          }
148
    }
149

    
150
  return((MYFLOAT)flx);
151
}
152

    
153
void impact(MYFLOAT d,MYFLOAT phi,int dim,MYFLOAT r,MYFLOAT b,MYFLOAT tho,MYFLOAT m,
154
            MYFLOAT **zp,MYFLOAT **fp,
155
            MYFLOAT q,MYFLOAT db,
156
            MYFLOAT h,MYFLOAT bss,int raie)
157
{
158
  MYFLOAT xe,ye;
159
  int xi,yi;
160
  MYFLOAT flx,rf;
161
  xe=d*sin(phi);
162
  ye=-d*cos(phi);
163

    
164
  xi=(int)xe+dim/2;
165
  yi=(int)ye+dim/2;
166

    
167
  rf=decalage_spectral(r,b,phi,tho,m);
168

    
169
  if (raie==0)
170
    {
171
      bss=1.e19;
172
      flx=spectre_cn(rf,b,db,h,r,m,bss);
173
    }
174
  else
175
    {
176
      bss=2.;
177
      flx=spectre(rf,q,b,db,h,r,m,bss);
178
    }
179
  
180
  if (zp[xi][yi]==0.)
181
    {
182
      zp[xi][yi]=1./rf;
183
    }
184
  
185
  if (fp[xi][yi]==0.)
186
    {
187
      fp[xi][yi]=flx;
188
    }
189

    
190
}
191

    
192
void sauvegarde_pgm(char nom[24],unsigned int **image,int dim)
193
{
194
  FILE            *sortie;
195
  unsigned long   i,j;
196
  
197
  sortie=fopen(nom,"w");
198
  
199
  fprintf(sortie,"P5\n");
200
  fprintf(sortie,"%i %i\n",dim,dim);
201
  fprintf(sortie,"255\n");
202

    
203
  for (j=0;j<dim;j++) for (i=0;i<dim;i++)
204
    {
205
      fputc(image[i][j],sortie);
206
    }
207

    
208
  fclose(sortie);
209
}
210

    
211
int main(int argc,char *argv[])
212
{
213

    
214
  MYFLOAT m,rs,ri,re,tho;
215
  int q;
216

    
217
  MYFLOAT bss,stp;
218
  int nmx,dim;
219
  MYFLOAT d,bmx,db,b,h;
220
  MYFLOAT up,vp,pp;
221
  MYFLOAT us,vs,ps;
222
  MYFLOAT rp[TRACKPOINTS];
223
  MYFLOAT **zp,**fp;
224
  unsigned int **izp,**ifp;
225
  MYFLOAT zmx,fmx,zen,fen;
226
  int zimx=0,zjmx=0,fimx=0,fjmx=0;
227
  MYFLOAT flux_tot,impc_tot;
228
  MYFLOAT phi,thi,thx,phd,php,nr,r;
229
  int ni,ii,i,imx,j,n,tst,dist,raie,pc,fcl,zcl;
230
  MYFLOAT nh;
231
  struct timeval tv1,tv2;
232
  struct timezone tz;  
233
  double elapsed;
234
  int mtv1,mtv2;
235
  
236
  if (argc==2)
237
    {
238
      if (strcmp(argv[1],"-aide")==0)
239
        {
240
          printf("\nSimulation d'un disque d'accretion autour d'un trou noir\n");
241
          printf("\nParametres a definir :\n\n");
242
          printf("  1) Dimension de l'Image\n");
243
          printf("  2) Masse relative du trou noir\n");
244
          printf("  3) Dimension du disque exterieur\n");
245
          printf("  4) Inclinaison par rapport au disque (en degres)\n");
246
          printf("  5) Rayonnement de disque MONOCHROMATIQUE ou CORPS_NOIR\n");
247
          printf("  6) Impression des images NEGATIVE ou POSITIVE\n"); 
248
          printf("  7) Nom de l'image des Flux\n");
249
          printf("  8) Nom de l'image des decalages spectraux\n");
250
          printf("\nSi aucun parametre defini, parametres par defaut :\n\n");
251
          printf("  1) Dimension de l'image : 1024 pixels de cote\n");
252
          printf("  2) Masse relative du trou noir : 1\n");
253
          printf("  3) Dimension du disque exterieur : 12 \n");
254
          printf("  4) Inclinaison par rapport au disque (en degres) : 10\n");
255
          printf("  5) Rayonnement de disque CORPS_NOIR\n");
256
          printf("  6) Impression des images NEGATIVE ou POSITIVE\n"); 
257
                 printf("  7) Nom de l'image des flux : flux.pgm\n");
258
          printf("  8) Nom de l'image des z : z.pgm\n");
259
        }
260
    }
261
  
262
  if (argc==9)
263
    {
264
      printf("# Utilisation les valeurs definies par l'utilisateur\n");
265
      
266
      dim=atoi(argv[1]);
267
      m=atof(argv[2]);
268
      re=atof(argv[3]);
269
      tho=PI/180.*(90-atof(argv[4]));
270
      
271
      rs=2.*m;
272
      ri=3.*rs;
273

    
274
      if (strcmp(argv[5],"CORPS_NOIR")==0)
275
        {
276
          raie=0;
277
        }
278
      else
279
        {
280
          raie=1;
281
        }
282

    
283
    }
284
  else
285
    {
286
      printf("# Utilisation les valeurs par defaut\n");
287
      
288
      dim=1024;
289
      m=1.;
290
      rs=2.*m;
291
      ri=3.*rs;
292
      re=12.;
293
      tho=PI/180.*80;
294
      // Corps noir
295
      raie=0;
296
    }
297

    
298
  if (raie==1)
299
    {
300
      bss=2.;
301
      q=-2;
302
    }
303
  else
304
    {
305
      bss=1.e19;
306
      q=-0.75;
307
    }
308

    
309
      printf("# Dimension de l'image : %i\n",dim);
310
      printf("# Masse : %f\n",m);
311
      printf("# Rayon singularite : %f\n",rs);
312
      printf("# Rayon interne : %f\n",ri);
313
      printf("# Rayon externe : %f\n",re);
314
      printf("# Inclinaison a la normale en radian : %f\n",tho);
315
  
316
  zp=(MYFLOAT**)calloc(dim,sizeof(MYFLOAT*));
317
  zp[0]=(MYFLOAT*)calloc(dim*dim,sizeof(MYFLOAT));
318
  
319
  fp=(MYFLOAT**)calloc(dim,sizeof(MYFLOAT*));
320
  fp[0]=(MYFLOAT*)calloc(dim*dim,sizeof(MYFLOAT));
321

    
322
  izp=(unsigned int**)calloc(dim,sizeof(unsigned int*));
323
  izp[0]=(unsigned int*)calloc(dim*dim,sizeof(unsigned int));
324
  
325
  ifp=(unsigned int**)calloc(dim,sizeof(unsigned int*));
326
  ifp[0]=(unsigned int*)calloc(dim*dim,sizeof(unsigned int));
327

    
328
  for (i=1;i<dim;i++)
329
    {
330
      zp[i]=zp[i-1]+dim;
331
      fp[i]=fp[i-1]+dim;
332
      izp[i]=izp[i-1]+dim;
333
      ifp[i]=ifp[i-1]+dim;
334
    }
335

    
336
  nmx=dim;
337
  stp=dim/(2.*nmx);
338
  bmx=1.25*re;
339
  b=0.;
340
  pc=0;
341

    
342
  // Set start timer
343
  gettimeofday(&tv1, &tz);
344
  //
345
  mtv1=clock()*1000/CLOCKS_PER_SEC;
346
  
347
  for (n=1;n<=nmx;n++)
348
    {     
349
      h=4.*PI/(MYFLOAT)TRACKPOINTS;
350
      d=stp*n;
351

    
352
      db=bmx/(MYFLOAT)nmx;
353
      b=db*(MYFLOAT)n;
354
      up=0.;
355
      vp=1.;
356
      
357
      pp=0.;
358
      nh=1;
359

    
360
      rungekutta(&ps,&us,&vs,pp,up,vp,h,m,b);
361
    
362
      rp[(int)nh]=fabs(b/us);
363
      
364
      do
365
        {
366
          nh++;
367
          pp=ps;
368
          up=us;
369
          vp=vs;
370
          rungekutta(&ps,&us,&vs,pp,up,vp,h,m,b);
371
          
372
          rp[(int)nh]=b/us;
373
          
374
        } while ((rp[(int)nh]>=rs)&&(rp[(int)nh]<=rp[1]));
375
      
376
      for (i=nh+1;i<TRACKPOINTS;i++)
377
        {
378
          rp[i]=0.; 
379
        }
380
      
381
      imx=(int)(8*d);
382
      
383
      for (i=0;i<=imx;i++)
384
        {
385
          phi=2.*PI/(MYFLOAT)imx*(MYFLOAT)i;
386
          phd=atanp(cos(phi)*sin(tho),cos(tho));
387
          phd=fmod(phd,PI);
388
          ii=0;
389
          tst=0;
390
          
391
          do
392
            {
393
              php=phd+(MYFLOAT)ii*PI;
394
              nr=php/h;
395
              ni=(int)nr;
396

    
397
              if ((MYFLOAT)ni<nh)
398
                {
399
                  r=(rp[ni+1]-rp[ni])*(nr-ni*1.)+rp[ni];
400
                }
401
              else
402
                {
403
                  r=rp[ni];
404
                }
405
           
406
              if ((r<=re)&&(r>=ri))
407
                {
408
                  tst=1;
409
                  impact(d,phi,dim,r,b,tho,m,zp,fp,q,db,h,bss,raie);
410
                }
411
              
412
              ii++;
413
            } while ((ii<=2)&&(tst==0));
414
        }
415
    }
416

    
417
  // Set stop timer
418
  gettimeofday(&tv2, &tz);
419
  mtv2=clock()*1000/CLOCKS_PER_SEC;
420
  
421
  //  elapsed=(double)((tv2.tv_sec-tv1.tv_sec) * 1000000L +
422
  //                          (tv2.tv_usec-tv1.tv_usec))/1000000;  
423
  elapsed=(double)((mtv2-mtv1)/1000.);  
424
  
425
  fmx=fp[0][0];
426
  zmx=zp[0][0];
427
  
428
  for (i=0;i<dim;i++) for (j=0;j<dim;j++)
429
    {
430
      if (fmx<fp[i][j])
431
        {
432
          fimx=i;
433
          fjmx=j;
434
          fmx=fp[i][j];
435
        }
436
      
437
      if (zmx<zp[i][j])
438
        {
439
          zimx=i;
440
          zjmx=j;
441
          zmx=zp[i][j];
442
        }
443
    }
444

    
445
  printf("\nElapsed Time : %lf",(double)elapsed);
446
  printf("\nZ max @(%i,%i) : %f",zimx,zjmx,zmx);
447
  printf("\nFlux max @(%i,%i) : %f\n\n",fimx,fjmx,fmx);
448

    
449
  for (i=0;i<dim;i++) for (j=0;j<dim;j++)
450
    {
451
      zcl=(int)(255/zmx*zp[i][dim-1-j]);
452
      fcl=(int)(255/fmx*fp[i][dim-1-j]);
453

    
454
      if (strcmp(argv[6],"NEGATIVE")==0)
455
        {
456
          if (zcl>255)
457
            {
458
              izp[i][j]=0;
459
            }
460
          else
461
            {
462
              izp[i][j]=255-zcl;
463
            }
464
          
465
          if (fcl>255)
466
            {
467
              ifp[i][j]=0;
468
            }
469
          else
470
            {
471
              ifp[i][j]=255-fcl;
472
            } 
473
          
474
        }
475
      else
476
        {
477
          if (zcl>255)
478
            {
479
              izp[i][j]=255;
480
            }
481
          else
482
            {
483
              izp[i][j]=zcl;
484
            }
485
          
486
          if (fcl>255)
487
            {
488
              ifp[i][j]=255;
489
            }
490
          else
491
            {
492
              ifp[i][j]=fcl;
493
            } 
494
          
495
        }
496
        
497
    }
498

    
499
  if (argc==9)
500
   {
501
     sauvegarde_pgm(argv[7],ifp,dim);
502
     sauvegarde_pgm(argv[8],izp,dim);
503
   }
504
  else
505
    {
506
      sauvegarde_pgm("z.pgm",izp,dim);
507
      sauvegarde_pgm("flux.pgm",ifp,dim);
508
    }
509

    
510
  free(zp[0]);
511
  free(zp);
512
  free(fp[0]);
513
  free(fp);
514

    
515
  free(izp[0]);
516
  free(izp);
517
  free(ifp[0]);
518
  free(ifp);
519

    
520
}
521

    
522