00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039
00040
00041
00042 #include "magick/studio.h"
00043 #include "magick/artifact.h"
00044 #include "magick/blob.h"
00045 #include "magick/cache.h"
00046 #include "magick/cache-view.h"
00047 #include "magick/color.h"
00048 #include "magick/color-private.h"
00049 #include "magick/draw.h"
00050 #include "magick/exception.h"
00051 #include "magick/exception-private.h"
00052 #include "magick/gem.h"
00053 #include "magick/image.h"
00054 #include "magick/image-private.h"
00055 #include "magick/list.h"
00056 #include "magick/memory_.h"
00057 #include "magick/pixel-private.h"
00058 #include "magick/property.h"
00059 #include "magick/monitor.h"
00060 #include "magick/monitor-private.h"
00061 #include "magick/pixel.h"
00062 #include "magick/option.h"
00063 #include "magick/resample.h"
00064 #include "magick/resize.h"
00065 #include "magick/resize-private.h"
00066 #include "magick/string_.h"
00067 #include "magick/utility.h"
00068 #include "magick/version.h"
00069 #if defined(MAGICKCORE_LQR_DELEGATE)
00070 #include <lqr.h>
00071 #endif
00072
00073
00074
00075
00076 struct _ResizeFilter
00077 {
00078 MagickRealType
00079 (*filter)(const MagickRealType,const ResizeFilter *),
00080 (*window)(const MagickRealType,const ResizeFilter *),
00081 support,
00082 window_support,
00083 scale,
00084 blur,
00085 cubic[8];
00086
00087 unsigned long
00088 signature;
00089 };
00090
00091
00092
00093
00094 static MagickRealType
00095 I0(MagickRealType x),
00096 BesselOrderOne(MagickRealType);
00097
00098
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119
00120
00121
00122
00123
00124
00125
00126
00127
00128
00129
00130 static MagickRealType Bessel(const MagickRealType x,
00131 const ResizeFilter *magick_unused(resize_filter))
00132 {
00133
00134
00135
00136
00137
00138
00139
00140
00141 if (x == 0.0)
00142 return((MagickRealType) (MagickPI/4.0));
00143 return(BesselOrderOne(MagickPI*x)/(2.0*x));
00144 }
00145
00146 static MagickRealType Blackman(const MagickRealType x,
00147 const ResizeFilter *magick_unused(resize_filter))
00148 {
00149
00150
00151
00152 return(0.42+0.5*cos(MagickPI*(double) x)+0.08*cos(2.0*MagickPI*(double) x));
00153 }
00154
00155 static MagickRealType Bohman(const MagickRealType x,
00156 const ResizeFilter *magick_unused(resize_filter))
00157 {
00158
00159
00160
00161 return((1-x)*cos(MagickPI*(double) x)+sin(MagickPI*(double) x)/MagickPI);
00162 }
00163
00164 static MagickRealType Box(const MagickRealType magick_unused(x),
00165 const ResizeFilter *magick_unused(resize_filter))
00166 {
00167
00168
00169
00170 return(1.0);
00171 }
00172
00173 static MagickRealType CubicBC(const MagickRealType x,
00174 const ResizeFilter *resize_filter)
00175 {
00176
00177
00178
00179
00180
00181
00182
00183
00184
00185
00186
00187
00188
00189
00190
00191
00192
00193
00194
00195
00196
00197
00198
00199
00200
00201
00202
00203
00204
00205
00206 if (x < 1.0)
00207 return(resize_filter->cubic[0]+x*(resize_filter->cubic[1]+x*
00208 (resize_filter->cubic[2]+x*resize_filter->cubic[3])));
00209 if (x < 2.0)
00210 return(resize_filter->cubic[4] +x*(resize_filter->cubic[5]+x*
00211 (resize_filter->cubic[6] +x*resize_filter->cubic[7])));
00212 return(0.0);
00213 }
00214
00215 static MagickRealType Gaussian(const MagickRealType x,
00216 const ResizeFilter *magick_unused(resize_filter))
00217 {
00218 return(exp((double) (-2.0*x*x))*sqrt(2.0/MagickPI));
00219 }
00220
00221 static MagickRealType Hanning(const MagickRealType x,
00222 const ResizeFilter *magick_unused(resize_filter))
00223 {
00224
00225
00226
00227 return(0.5+0.5*cos(MagickPI*(double) x));
00228 }
00229
00230 static MagickRealType Hamming(const MagickRealType x,
00231 const ResizeFilter *magick_unused(resize_filter))
00232 {
00233
00234
00235
00236 return(0.54+0.46*cos(MagickPI*(double) x));
00237 }
00238
00239 static MagickRealType Kaiser(const MagickRealType x,
00240 const ResizeFilter *magick_unused(resize_filter))
00241 {
00242 #define Alpha 6.5
00243 #define I0A (1.0/I0(Alpha))
00244
00245
00246
00247
00248
00249
00250 return(I0A*I0(Alpha*sqrt((double) (1.0-x*x))));
00251 }
00252
00253 static MagickRealType Lagrange(const MagickRealType x,
00254 const ResizeFilter *resize_filter)
00255 {
00256 long
00257 n,
00258 order;
00259
00260 MagickRealType
00261 value;
00262
00263 register long
00264 i;
00265
00266
00267
00268
00269
00270
00271
00272
00273
00274
00275
00276
00277 if (x > resize_filter->support)
00278 return(0.0);
00279 order=(long) (2.0*resize_filter->window_support);
00280 n=(long) ((1.0*order)/2.0+x);
00281 value=1.0f;
00282 for (i=0; i < order; i++)
00283 if (i != n)
00284 value*=(n-i-x)/(n-i);
00285 return(value);
00286 }
00287
00288 static MagickRealType Quadratic(const MagickRealType x,
00289 const ResizeFilter *magick_unused(resize_filter))
00290 {
00291
00292
00293
00294 if (x < 0.5)
00295 return(0.75-x*x);
00296 if (x < 1.5)
00297 return(0.5*(x-1.5)*(x-1.5));
00298 return(0.0);
00299 }
00300
00301 static MagickRealType Sinc(const MagickRealType x,
00302 const ResizeFilter *magick_unused(resize_filter))
00303 {
00304
00305
00306
00307 if (x == 0.0)
00308 return(1.0);
00309 return(sin(MagickPI*(double) x)/(MagickPI*(double) x));
00310 }
00311
00312 static MagickRealType Triangle(const MagickRealType x,
00313 const ResizeFilter *magick_unused(resize_filter))
00314 {
00315
00316
00317
00318
00319 if (x < 1.0)
00320 return(1.0-x);
00321 return(0.0);
00322 }
00323
00324 static MagickRealType Welsh(const MagickRealType x,
00325 const ResizeFilter *magick_unused(resize_filter))
00326 {
00327
00328
00329
00330 if (x < 1.0)
00331 return(1.0-x*x);
00332 return(0.0);
00333 }
00334
00335
00336
00337
00338
00339
00340
00341
00342
00343
00344
00345
00346
00347
00348
00349
00350
00351
00352
00353
00354
00355
00356
00357
00358
00359
00360
00361
00362
00363
00364
00365
00366
00367
00368
00369
00370
00371
00372
00373
00374
00375
00376
00377
00378
00379
00380
00381
00382
00383
00384
00385
00386
00387
00388
00389
00390
00391
00392
00393
00394
00395
00396
00397
00398
00399
00400
00401
00402
00403
00404
00405
00406
00407
00408
00409
00410
00411
00412
00413
00414
00415
00416
00417
00418
00419
00420
00421
00422
00423
00424
00425
00426
00427
00428
00429
00430
00431
00432
00433
00434
00435
00436
00437
00438
00439
00440
00441
00442
00443
00444
00445
00446
00447
00448
00449
00450
00451
00452
00453 MagickExport ResizeFilter *AcquireResizeFilter(const Image *image,
00454 const FilterTypes filter, const MagickRealType blur,
00455 const MagickBooleanType cylindrical,ExceptionInfo *exception)
00456 {
00457 const char
00458 *artifact;
00459
00460 FilterTypes
00461 filter_type,
00462 window_type;
00463
00464 long
00465 filter_artifact;
00466
00467 MagickRealType
00468 B,
00469 C;
00470
00471 register ResizeFilter
00472 *resize_filter;
00473
00474
00475
00476
00477
00478
00479
00480
00481 static struct
00482 {
00483 FilterTypes
00484 filter,
00485 window;
00486 } const mapping[SentinelFilter] =
00487 {
00488 { UndefinedFilter, BoxFilter },
00489 { PointFilter, BoxFilter },
00490 { BoxFilter, BoxFilter },
00491 { TriangleFilter, BoxFilter },
00492 { HermiteFilter, BoxFilter },
00493 { SincFilter, HanningFilter },
00494 { SincFilter, HammingFilter },
00495 { SincFilter, BlackmanFilter },
00496 { GaussianFilter, BoxFilter },
00497 { QuadraticFilter, BoxFilter },
00498 { CubicFilter, BoxFilter },
00499 { CatromFilter, BoxFilter },
00500 { MitchellFilter, BoxFilter },
00501 { LanczosFilter, SincFilter },
00502 { BesselFilter, BlackmanFilter },
00503 { SincFilter, BlackmanFilter },
00504 { SincFilter, KaiserFilter },
00505 { SincFilter, WelshFilter },
00506 { SincFilter, CubicFilter },
00507 { LagrangeFilter, BoxFilter },
00508 { SincFilter, BohmanFilter },
00509 { SincFilter, TriangleFilter }
00510 };
00511
00512
00513
00514
00515
00516
00517 static struct
00518 {
00519 MagickRealType
00520 (*function)(const MagickRealType, const ResizeFilter*),
00521 support,
00522 scale,
00523 B,
00524 C;
00525 } const filters[SentinelFilter] =
00526 {
00527 { Box, 0.0f, 0.5f, 0.0f, 0.0f },
00528 { Box, 0.0f, 0.5f, 0.0f, 0.0f },
00529 { Box, 0.5f, 0.5f, 0.0f, 0.0f },
00530 { Triangle, 1.0f, 1.0f, 0.0f, 0.0f },
00531 { CubicBC, 1.0f, 1.0f, 0.0f, 0.0f },
00532 { Hanning, 1.0f, 1.0f, 0.0f, 0.0f },
00533 { Hamming, 1.0f, 1.0f, 0.0f, 0.0f },
00534 { Blackman, 1.0f, 1.0f, 0.0f, 0.0f },
00535 { Gaussian, 1.5f, 1.5f, 0.0f, 0.0f },
00536 { Quadratic, 1.5f, 1.5f, 0.0f, 0.0f },
00537 { CubicBC, 2.0f, 2.0f, 1.0f, 0.0f },
00538 { CubicBC, 2.0f, 1.0f, 0.0f, 0.5f },
00539 { CubicBC, 2.0f, 1.0f, 1.0f/3.0f, 1.0f/3.0f },
00540 { Sinc, 3.0f, 1.0f, 0.0f, 0.0f },
00541 { Bessel, 3.2383f,1.2197f,.0f,.0f },
00542 { Sinc, 4.0f, 1.0f, 0.0f, 0.0f },
00543 { Kaiser, 1.0f, 1.0f, 0.0f, 0.0f },
00544 { Welsh, 1.0f, 1.0f, 0.0f, 0.0f },
00545 { CubicBC, 2.0f, 2.0f, 1.0f, 0.0f },
00546 { Lagrange, 2.0f, 1.0f, 0.0f, 0.0f },
00547 { Bohman, 1.0f, 1.0f, 0.0f, 0.0f },
00548 { Triangle, 1.0f, 1.0f, 0.0f, 0.0f }
00549 };
00550
00551
00552
00553
00554
00555
00556 static MagickRealType
00557 bessel_zeros[16] =
00558 {
00559 1.21966989126651f,
00560 2.23313059438153f,
00561 3.23831548416624f,
00562 4.24106286379607f,
00563 5.24276437687019f,
00564 6.24392168986449f,
00565 7.24475986871996f,
00566 8.24539491395205f,
00567 9.24589268494948f,
00568 10.2462933487549f,
00569 11.2466227948779f,
00570 12.2468984611381f,
00571 13.2471325221811f,
00572 14.2473337358069f,
00573 15.2475085630373f,
00574 16.247661874701f
00575 };
00576
00577 assert(image != (const Image *) NULL);
00578 assert(image->signature == MagickSignature);
00579 if (image->debug != MagickFalse)
00580 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
00581 assert(UndefinedFilter < filter && filter < SentinelFilter);
00582 assert(exception != (ExceptionInfo *) NULL);
00583 assert(exception->signature == MagickSignature);
00584
00585 resize_filter=(ResizeFilter *) AcquireMagickMemory(sizeof(*resize_filter));
00586 if (resize_filter == (ResizeFilter *) NULL)
00587 ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
00588
00589
00590 filter_type = mapping[filter].filter;
00591 window_type = mapping[filter].window;
00592
00593
00594
00595 resize_filter->blur = blur;
00596 artifact=GetImageArtifact(image,"filter:blur");
00597 if (artifact != (const char *) NULL)
00598 resize_filter->blur = atof(artifact);
00599 if ( resize_filter->blur < MagickEpsilon )
00600 resize_filter->blur = (MagickRealType) MagickEpsilon;
00601
00602
00603 if ( cylindrical != MagickFalse && filter != SincFilter ) {
00604
00605 if ( filter_type == SincFilter )
00606 filter_type = BesselFilter;
00607
00608 else if ( filter_type == LanczosFilter ) {
00609 filter_type = BesselFilter;
00610 window_type = BesselFilter;
00611 }
00612
00613 else if ( filter_type == GaussianFilter )
00614
00615
00616 resize_filter->blur *= 2.0*log(2.0)/sqrt(2.0/MagickPI);
00617 else if ( filter_type != BesselFilter )
00618
00619 resize_filter->blur *= bessel_zeros[0];
00620 }
00621
00622
00623 artifact=GetImageArtifact(image,"filter:filter");
00624 if (artifact != (const char *) NULL) {
00625
00626 filter_artifact=ParseMagickOption(MagickFilterOptions,
00627 MagickFalse,artifact);
00628 if ( UndefinedFilter < filter_artifact &&
00629 filter_artifact < SentinelFilter ) {
00630 filter_type = (FilterTypes) filter_artifact;
00631 window_type = BoxFilter;
00632 }
00633
00634 if ( filter_artifact == LanczosFilter ) {
00635 filter_type = (cylindrical!=MagickFalse) ? BesselFilter : LanczosFilter;
00636 window_type = (cylindrical!=MagickFalse) ? BesselFilter : SincFilter;
00637 }
00638
00639 artifact=GetImageArtifact(image,"filter:window");
00640 if (artifact != (const char *) NULL) {
00641 filter_artifact=ParseMagickOption(MagickFilterOptions,
00642 MagickFalse,artifact);
00643 if ( UndefinedFilter < filter_artifact &&
00644 filter_artifact < SentinelFilter ) {
00645 if ( filter_artifact != LanczosFilter )
00646 window_type = (FilterTypes) filter_artifact;
00647 else
00648 window_type = (cylindrical!=MagickFalse) ? BesselFilter : SincFilter;
00649 }
00650 }
00651 }
00652 else {
00653
00654 artifact=GetImageArtifact(image,"filter:window");
00655 if (artifact != (const char *) NULL) {
00656 filter_artifact=ParseMagickOption(MagickFilterOptions,MagickFalse,
00657 artifact);
00658 if ( UndefinedFilter < filter_artifact &&
00659 filter_artifact < SentinelFilter ) {
00660 filter_type = (cylindrical!=MagickFalse) ? BesselFilter : SincFilter;
00661 if ( filter_artifact != LanczosFilter )
00662 window_type = (FilterTypes) filter_artifact;
00663 else
00664 window_type = filter_type;
00665 }
00666 }
00667 }
00668
00669 resize_filter->filter = filters[filter_type].function;
00670 resize_filter->support = filters[filter_type].support;
00671 resize_filter->window = filters[window_type].function;
00672 resize_filter->scale = filters[window_type].scale;
00673 resize_filter->signature=MagickSignature;
00674
00675
00676 artifact=GetImageArtifact(image,"filter:lobes");
00677 if (artifact != (const char *) NULL) {
00678 long lobes = atol(artifact);
00679 if ( lobes < 1 ) lobes = 1;
00680 resize_filter->support = (MagickRealType) lobes;
00681 if ( filter_type == BesselFilter ) {
00682 if ( lobes > 16 ) lobes = 16;
00683 resize_filter->support = bessel_zeros[lobes-1];
00684 }
00685 }
00686 artifact=GetImageArtifact(image,"filter:support");
00687 if (artifact != (const char *) NULL)
00688 resize_filter->support = fabs(atof(artifact));
00689
00690
00691
00692
00693 resize_filter->window_support = resize_filter->support;
00694 artifact=GetImageArtifact(image,"filter:win-support");
00695 if (artifact != (const char *) NULL)
00696 resize_filter->window_support = fabs(atof(artifact));
00697
00698
00699 B=0.0;
00700 C=0.0;
00701 if ( filters[filter_type].function == CubicBC
00702 || filters[window_type].function == CubicBC ) {
00703 if ( filters[filter_type].function == CubicBC ) {
00704 B=filters[filter_type].B;
00705 C=filters[filter_type].C;
00706 }
00707 else if ( filters[window_type].function == CubicBC ) {
00708 B=filters[window_type].B;
00709 C=filters[window_type].C;
00710 }
00711 artifact=GetImageArtifact(image,"filter:b");
00712 if (artifact != (const char *) NULL) {
00713 B=atof(artifact);
00714 C=1.0-2.0*B;
00715 artifact=GetImageArtifact(image,"filter:c");
00716 if (artifact != (const char *) NULL)
00717 C=atof(artifact);
00718 }
00719 else {
00720 artifact=GetImageArtifact(image,"filter:c");
00721 if (artifact != (const char *) NULL) {
00722 C=atof(artifact);
00723 B=(1.0-C)/2.0;
00724 }
00725 }
00726
00727 resize_filter->cubic[0]=( 6.0 -2.0*B )/6.0;
00728 resize_filter->cubic[1]=0.0;
00729 resize_filter->cubic[2]=(-18.0+12.0*B+ 6.0*C)/6.0;
00730 resize_filter->cubic[3]=( 12.0- 9.0*B- 6.0*C)/6.0;
00731 resize_filter->cubic[4]=( 8.0*B+24.0*C)/6.0;
00732 resize_filter->cubic[5]=( -12.0*B-48.0*C)/6.0;
00733 resize_filter->cubic[6]=( 6.0*B+30.0*C)/6.0;
00734 resize_filter->cubic[7]=( - 1.0*B- 6.0*C)/6.0;
00735 }
00736 artifact=GetImageArtifact(image,"filter:verbose");
00737 if (artifact != (const char *) NULL)
00738 {
00739 double
00740 support,
00741 x;
00742
00743
00744
00745
00746 support=GetResizeFilterSupport(resize_filter);
00747 (void) printf("# support = %lg\n",support);
00748 for (x=0.0; x <= support; x+=0.01f)
00749 (void) printf("%5.2lf\t%lf\n",x,GetResizeFilterWeight(resize_filter,x));
00750 (void) printf("%5.2lf\t%lf\n",support,0.0);
00751 }
00752 return(resize_filter);
00753 }
00754
00755
00756
00757
00758
00759
00760
00761
00762
00763
00764
00765
00766
00767
00768
00769
00770
00771
00772
00773
00774
00775
00776
00777
00778
00779
00780
00781
00782
00783
00784
00785 MagickExport Image *AdaptiveResizeImage(const Image *image,
00786 const unsigned long columns,const unsigned long rows,ExceptionInfo *exception)
00787 {
00788 #define AdaptiveResizeImageTag "Resize/Image"
00789
00790 Image
00791 *resize_image;
00792
00793 long
00794 y;
00795
00796 MagickBooleanType
00797 proceed;
00798
00799 MagickPixelPacket
00800 pixel;
00801
00802 PointInfo
00803 offset;
00804
00805 register IndexPacket
00806 *resize_indexes;
00807
00808 register long
00809 x;
00810
00811 register PixelPacket
00812 *q;
00813
00814 ResampleFilter
00815 *resample_filter;
00816
00817 ViewInfo
00818 *resize_view;
00819
00820
00821
00822
00823 assert(image != (const Image *) NULL);
00824 assert(image->signature == MagickSignature);
00825 if (image->debug != MagickFalse)
00826 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
00827 assert(exception != (ExceptionInfo *) NULL);
00828 assert(exception->signature == MagickSignature);
00829 if ((columns == 0) || (rows == 0))
00830 return((Image *) NULL);
00831 if ((columns == image->columns) && (rows == image->rows))
00832 return(CloneImage(image,0,0,MagickTrue,exception));
00833 resize_image=CloneImage(image,columns,rows,MagickTrue,exception);
00834 if (resize_image == (Image *) NULL)
00835 return((Image *) NULL);
00836 if (SetImageStorageClass(resize_image,DirectClass) == MagickFalse)
00837 {
00838 InheritException(exception,&resize_image->exception);
00839 resize_image=DestroyImage(resize_image);
00840 return((Image *) NULL);
00841 }
00842 GetMagickPixelPacket(image,&pixel);
00843 resample_filter=AcquireResampleFilter(image,exception);
00844 if (image->interpolate == UndefinedInterpolatePixel)
00845 (void) SetResampleFilterInterpolateMethod(resample_filter,
00846 MeshInterpolatePixel);
00847 resize_view=AcquireCacheView(resize_image);
00848 for (y=0; y < (long) resize_image->rows; y++)
00849 {
00850 q=QueueCacheViewAuthenticPixels(resize_view,0,y,resize_image->columns,1,
00851 exception);
00852 if (q == (PixelPacket *) NULL)
00853 break;
00854 resize_indexes=GetCacheViewAuthenticIndexQueue(resize_view);
00855 offset.y=((MagickRealType) y*image->rows/resize_image->rows);
00856 for (x=0; x < (long) resize_image->columns; x++)
00857 {
00858 offset.x=((MagickRealType) x*image->columns/resize_image->columns);
00859 (void) ResamplePixelColor(resample_filter,offset.x-0.5,offset.y-0.5,
00860 &pixel);
00861 SetPixelPacket(resize_image,&pixel,q,resize_indexes+x);
00862 q++;
00863 }
00864 if (SyncCacheViewAuthenticPixels(resize_view,exception) == MagickFalse)
00865 break;
00866 proceed=SetImageProgress(image,AdaptiveResizeImageTag,y,image->rows);
00867 if (proceed == MagickFalse)
00868 break;
00869 }
00870 resample_filter=DestroyResampleFilter(resample_filter);
00871 resize_view=DestroyCacheView(resize_view);
00872 return(resize_image);
00873 }
00874
00875
00876
00877
00878
00879
00880
00881
00882
00883
00884
00885
00886
00887
00888
00889
00890
00891
00892
00893
00894
00895
00896
00897
00898
00899
00900
00901
00902
00903
00904
00905
00906
00907
00908
00909
00910
00911
00912
00913
00914 #undef I0
00915 static MagickRealType I0(MagickRealType x)
00916 {
00917 MagickRealType
00918 sum,
00919 t,
00920 y;
00921
00922 register long
00923 i;
00924
00925
00926
00927
00928 sum=1.0;
00929 y=x*x/4.0;
00930 t=y;
00931 for (i=2; t > MagickEpsilon; i++)
00932 {
00933 sum+=t;
00934 t*=y/((MagickRealType) i*i);
00935 }
00936 return(sum);
00937 }
00938
00939 #undef J1
00940 static MagickRealType J1(MagickRealType x)
00941 {
00942 MagickRealType
00943 p,
00944 q;
00945
00946 register long
00947 i;
00948
00949 static const double
00950 Pone[] =
00951 {
00952 0.581199354001606143928050809e+21,
00953 -0.6672106568924916298020941484e+20,
00954 0.2316433580634002297931815435e+19,
00955 -0.3588817569910106050743641413e+17,
00956 0.2908795263834775409737601689e+15,
00957 -0.1322983480332126453125473247e+13,
00958 0.3413234182301700539091292655e+10,
00959 -0.4695753530642995859767162166e+7,
00960 0.270112271089232341485679099e+4
00961 },
00962 Qone[] =
00963 {
00964 0.11623987080032122878585294e+22,
00965 0.1185770712190320999837113348e+20,
00966 0.6092061398917521746105196863e+17,
00967 0.2081661221307607351240184229e+15,
00968 0.5243710262167649715406728642e+12,
00969 0.1013863514358673989967045588e+10,
00970 0.1501793594998585505921097578e+7,
00971 0.1606931573481487801970916749e+4,
00972 0.1e+1
00973 };
00974
00975 p=Pone[8];
00976 q=Qone[8];
00977 for (i=7; i >= 0; i--)
00978 {
00979 p=p*x*x+Pone[i];
00980 q=q*x*x+Qone[i];
00981 }
00982 return(p/q);
00983 }
00984
00985 #undef P1
00986 static MagickRealType P1(MagickRealType x)
00987 {
00988 MagickRealType
00989 p,
00990 q;
00991
00992 register long
00993 i;
00994
00995 static const double
00996 Pone[] =
00997 {
00998 0.352246649133679798341724373e+5,
00999 0.62758845247161281269005675e+5,
01000 0.313539631109159574238669888e+5,
01001 0.49854832060594338434500455e+4,
01002 0.2111529182853962382105718e+3,
01003 0.12571716929145341558495e+1
01004 },
01005 Qone[] =
01006 {
01007 0.352246649133679798068390431e+5,
01008 0.626943469593560511888833731e+5,
01009 0.312404063819041039923015703e+5,
01010 0.4930396490181088979386097e+4,
01011 0.2030775189134759322293574e+3,
01012 0.1e+1
01013 };
01014
01015 p=Pone[5];
01016 q=Qone[5];
01017 for (i=4; i >= 0; i--)
01018 {
01019 p=p*(8.0/x)*(8.0/x)+Pone[i];
01020 q=q*(8.0/x)*(8.0/x)+Qone[i];
01021 }
01022 return(p/q);
01023 }
01024
01025 #undef Q1
01026 static MagickRealType Q1(MagickRealType x)
01027 {
01028 MagickRealType
01029 p,
01030 q;
01031
01032 register long
01033 i;
01034
01035 static const double
01036 Pone[] =
01037 {
01038 0.3511751914303552822533318e+3,
01039 0.7210391804904475039280863e+3,
01040 0.4259873011654442389886993e+3,
01041 0.831898957673850827325226e+2,
01042 0.45681716295512267064405e+1,
01043 0.3532840052740123642735e-1
01044 },
01045 Qone[] =
01046 {
01047 0.74917374171809127714519505e+4,
01048 0.154141773392650970499848051e+5,
01049 0.91522317015169922705904727e+4,
01050 0.18111867005523513506724158e+4,
01051 0.1038187585462133728776636e+3,
01052 0.1e+1
01053 };
01054
01055 p=Pone[5];
01056 q=Qone[5];
01057 for (i=4; i >= 0; i--)
01058 {
01059 p=p*(8.0/x)*(8.0/x)+Pone[i];
01060 q=q*(8.0/x)*(8.0/x)+Qone[i];
01061 }
01062 return(p/q);
01063 }
01064
01065 static MagickRealType BesselOrderOne(MagickRealType x)
01066 {
01067 MagickRealType
01068 p,
01069 q;
01070
01071 if (x == 0.0)
01072 return(0.0);
01073 p=x;
01074 if (x < 0.0)
01075 x=(-x);
01076 if (x < 8.0)
01077 return(p*J1(x));
01078 q=sqrt((double) (2.0/(MagickPI*x)))*(P1(x)*(1.0/sqrt(2.0)*(sin((double) x)-
01079 cos((double) x)))-8.0/x*Q1(x)*(-1.0/sqrt(2.0)*(sin((double) x)+
01080 cos((double) x))));
01081 if (p < 0.0)
01082 q=(-q);
01083 return(q);
01084 }
01085
01086
01087
01088
01089
01090
01091
01092
01093
01094
01095
01096
01097
01098
01099