cjpeg.txt
上传用户:wangchao
上传日期:2021-02-19
资源大小:9k
文件大小:31k
源码类别:

压缩解压

开发平台:

C++ Builder

  1. #ifndef __JPEGDEC_H__
  2. #define __JPEGDEC_H__ 
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #define BYTE unsigned char
  7. #define WORD unsigned short int
  8. #define DWORD unsigned int
  9. #define SDWORD signed int
  10. #define SBYTE signed char
  11. #define SWORD signed short int
  12. int load_JPEG_header(FILE *fp, DWORD *X_image, DWORD *Y_image);
  13. void decode_JPEG_image();
  14. int get_JPEG_buffer(WORD X_image,WORD Y_image, BYTE **address_dest_buffer);
  15. #endif
  16. #include <conio.h>
  17. #include <time.h>
  18. char *FileName="image.jpg";
  19. extern char error_string[90];
  20. typedef struct s_BM_header {
  21.   WORD BMP_id ; // 'B''M'
  22.   DWORD size; // size in bytes of the BMP file
  23.   DWORD zero_res; // 0
  24.   DWORD offbits; // 54
  25.   DWORD biSize; // 0x28
  26.   DWORD Width;  // X
  27.   DWORD Height;  // Y
  28.   WORD  biPlanes; // 1
  29.   WORD  biBitCount ; // 24
  30.   DWORD biCompression; // 0 = BI_RGB
  31.   DWORD biSizeImage; // 0
  32.   DWORD biXPelsPerMeter; // 0xB40
  33.   DWORD biYPelsPerMeter; // 0xB40
  34.   DWORD biClrUsed; //0
  35.   DWORD biClrImportant; //0
  36. } BM_header;
  37. typedef struct s_RGB {
  38.  BYTE B;
  39.      BYTE G;
  40.      BYTE R;
  41. } RGB;
  42. void exitmessage(char *message)
  43. {
  44.  printf("%sn",message);exit(0);
  45. }
  46. void write_buf_to_BMP(BYTE *im_buffer, WORD X_bitmap, WORD Y_bitmap, char *BMPname)
  47. {
  48.  SWORD x,y;
  49.  RGB *pixel;
  50.  BM_header BH;
  51.  FILE *fp_bitmap;
  52.  DWORD im_loc_bytes;
  53.  BYTE nr_fillingbytes, i;
  54.  BYTE zero_byte=0;
  55.  fp_bitmap=fopen(BMPname,"wb");
  56.  if (fp_bitmap==NULL) exitmessage("File cannot be created");
  57.  if (X_bitmap%4!=0) nr_fillingbytes=4-((X_bitmap*3L)%4);
  58.   else nr_fillingbytes=0;
  59.  BH.BMP_id = 'M'*256+'B';     fwrite(&BH.BMP_id,2,1,fp_bitmap);
  60.  BH.size=54+Y_bitmap*(X_bitmap*3+nr_fillingbytes);fwrite(&BH.size,4,1,fp_bitmap);
  61.  BH.zero_res = 0;             fwrite(&BH.zero_res,4,1,fp_bitmap);
  62.  BH.offbits = 54;             fwrite(&BH.offbits,4,1,fp_bitmap);
  63.  BH.biSize = 0x28;            fwrite(&BH.biSize,4,1,fp_bitmap);
  64.  BH.Width = X_bitmap;       fwrite(&BH.Width,4,1,fp_bitmap);
  65.  BH.Height = Y_bitmap;       fwrite(&BH.Height,4,1,fp_bitmap);
  66.  BH.biPlanes = 1;             fwrite(&BH.biPlanes,2,1,fp_bitmap);
  67.  BH.biBitCount = 24;          fwrite(&BH.biBitCount,2,1,fp_bitmap);
  68.  BH.biCompression = 0;        fwrite(&BH.biCompression,4,1,fp_bitmap);
  69.  BH.biSizeImage = 0;          fwrite(&BH.biSizeImage,4,1,fp_bitmap);
  70.  BH.biXPelsPerMeter = 0xB40;  fwrite(&BH.biXPelsPerMeter,4,1,fp_bitmap);
  71.  BH.biYPelsPerMeter = 0xB40;  fwrite(&BH.biYPelsPerMeter,4,1,fp_bitmap);
  72.  BH.biClrUsed = 0;           fwrite(&BH.biClrUsed,4,1,fp_bitmap);
  73.  BH.biClrImportant = 0;       fwrite(&BH.biClrImportant,4,1,fp_bitmap);
  74.  printf("Writing bitmap ...n");
  75.  im_loc_bytes=(DWORD)im_buffer+((DWORD)Y_bitmap-1)*X_bitmap*4;
  76.  for (y=0;y<Y_bitmap;y++)
  77.   {
  78.    for (x=0;x<X_bitmap;x++)
  79. {
  80.  pixel=(RGB *)im_loc_bytes;
  81.  fwrite(pixel, 3, 1, fp_bitmap);
  82.  im_loc_bytes+=4;
  83. }
  84.    for (i=0;i<nr_fillingbytes;i++)
  85.    fwrite(&zero_byte,1,1,fp_bitmap);
  86.    im_loc_bytes-=2L*X_bitmap*4;
  87.   }
  88.  printf("Done.n");
  89.  fclose(fp_bitmap);
  90. }
  91. // Used markers:
  92. #define SOI 0xD8
  93. #define EOI 0xD9
  94. #define APP0 0xE0
  95. #define SOF 0xC0
  96. #define DQT 0xDB
  97. #define DHT 0xC4
  98. #define SOS 0xDA
  99. #define DRI 0xDD
  100. #define COM 0xFE
  101. char error_string[90];
  102. #define exit_func(err) { strcpy(error_string, err); return 0;}
  103. static BYTE *buf; // the buffer we use for storing the entire JPG file
  104. static BYTE bp; //current byte
  105. static WORD wp; //current word
  106. static DWORD byte_pos; // current byte position
  107. #define BYTE_p(i) bp=buf[(i)++]
  108. #define WORD_p(i) wp=(((WORD)(buf[(i)]))<<8) + buf[(i)+1]; (i)+=2
  109. // WORD X_image_size,Y_image_size; // X,Y sizes of the image
  110. static WORD X_round,Y_round; // The dimensions rounded to multiple of Hmax*8 (X_round)
  111.   // and Ymax*8 (Y_round)
  112. static BYTE *im_buffer; // RGBA image buffer
  113. static DWORD X_image_bytes; // size in bytes of 1 line of the image = X_round * 4
  114. static DWORD y_inc_value ; // 32*X_round; // used by decode_MCU_1x2,2x1,2x2
  115. BYTE YH,YV,CbH,CbV,CrH,CrV; // sampling factors (horizontal and vertical) for Y,Cb,Cr
  116. static WORD Hmax,Vmax;
  117. static BYTE zigzag[64]={ 0, 1, 5, 6,14,15,27,28,
  118.   2, 4, 7,13,16,26,29,42,
  119.   3, 8,12,17,25,30,41,43,
  120.   9,11,18,24,31,40,44,53,
  121.  10,19,23,32,39,45,52,54,
  122.  20,22,33,38,46,51,55,60,
  123.  21,34,37,47,50,56,59,61,
  124.  35,36,48,49,57,58,62,63 };
  125. typedef struct {
  126.    BYTE Length[17];  // k =1-16 ; L[k] indicates the number of Huffman codes of length k
  127.    WORD minor_code[17];  // indicates the value of the smallest Huffman code of length k
  128.    WORD major_code[17];  // similar, but the highest code
  129.    BYTE V[65536];  // V[k][j] = Value associated to the j-th Huffman code of length k
  130. // High nibble = nr of previous 0 coefficients
  131. // Low nibble = size (in bits) of the coefficient which will be taken from the data stream
  132. } Huffman_table;
  133. static float *QT[4]; // quantization tables, no more than 4 quantization tables (QT[0..3])
  134. static Huffman_table HTDC[4]; //DC huffman tables , no more than 4 (0..3)
  135. static Huffman_table HTAC[4]; //AC huffman tables                  (0..3)
  136. static BYTE YQ_nr,CbQ_nr,CrQ_nr; // quantization table number for Y, Cb, Cr
  137. static BYTE YDC_nr,CbDC_nr,CrDC_nr; // DC Huffman table number for Y,Cb, Cr
  138. static BYTE YAC_nr,CbAC_nr,CrAC_nr; // AC Huffman table number for Y,Cb, Cr
  139. static BYTE Restart_markers; // if 1 => Restart markers on , 0 => no restart markers
  140. static WORD MCU_restart; //Restart markers appears every MCU_restart MCU blocks
  141. typedef void (*decode_MCU_func)(DWORD);
  142. static SWORD DCY, DCCb, DCCr; // Coeficientii DC pentru Y,Cb,Cr
  143. static SWORD DCT_coeff[64]; // Current DCT_coefficients
  144. static BYTE Y[64],Cb[64],Cr[64]; // Y, Cb, Cr of the current 8x8 block for the 1x1 case
  145. static BYTE Y_1[64],Y_2[64],Y_3[64],Y_4[64];
  146. static BYTE tab_1[64],tab_2[64],tab_3[64],tab_4[64]; // tabelele de supraesantionare pt cele 4 blocuri
  147. static SWORD Cr_tab[256],Cb_tab[256]; // Precalculated Cr, Cb tables
  148. static SWORD Cr_Cb_green_tab[65536];
  149. // Initial conditions:
  150. // byte_pos = start position in the Huffman coded segment
  151. // WORD_get(w1); WORD_get(w2);wordval=w1;
  152. static BYTE d_k=0;  // Bit displacement in memory, relative to the offset of w1
  153.  // it's always <16
  154. static WORD w1,w2; // w1 = First word in memory; w2 = Second word
  155. static DWORD wordval ; // the actual used value in Huffman decoding.
  156. static DWORD mask[17];
  157. static SWORD neg_pow2[17]={0,-1,-3,-7,-15,-31,-63,-127,-255,-511,-1023,-2047,-4095,-8191,-16383,-32767};
  158. static DWORD start_neg_pow2=(DWORD)neg_pow2;
  159. static int shift_temp;
  160. #define RIGHT_SHIFT(x,shft)  
  161. ((shift_temp = (x)) < 0 ? 
  162.  (shift_temp >> (shft)) | ((~(0L)) << (32-(shft))) : 
  163.  (shift_temp >> (shft)))
  164. #define DESCALE(x,n)  RIGHT_SHIFT((x) + (1L << ((n)-1)), n)
  165. #define RANGE_MASK 1023L
  166. static BYTE *rlimit_table;
  167. void prepare_range_limit_table()
  168. /* Allocate and fill in the sample_range_limit table */
  169. {
  170.   int j;
  171.   rlimit_table = (BYTE *)malloc(5 * 256L + 128) ;
  172.   /* First segment of "simple" table: limit[x] = 0 for x < 0 */
  173.   memset((void *)rlimit_table,0,256);
  174.   rlimit_table += 256; /* allow negative subscripts of simple table */
  175.   /* Main part of "simple" table: limit[x] = x */
  176.   for (j = 0; j < 256; j++) rlimit_table[j] = j;
  177.   /* End of simple table, rest of first half of post-IDCT table */
  178.   for (j = 256; j < 640; j++) rlimit_table[j] = 255;
  179.   /* Second half of post-IDCT table */
  180.   memset((void *)(rlimit_table + 640),0,384);
  181.   for (j = 0; j < 128 ; j++) rlimit_table[j+1024] = j;
  182. }
  183. #ifdef _MSC_VER
  184. WORD lookKbits(BYTE k)
  185. {
  186.  _asm {
  187.  mov dl, k
  188.  mov cl, 16
  189.  sub cl, dl
  190.  mov eax, [wordval]
  191.  shr eax, cl
  192.  }
  193. }
  194. WORD WORD_hi_lo(BYTE byte_high,BYTE byte_low)
  195. {
  196. _asm {
  197.   mov ah,byte_high
  198.   mov al,byte_low
  199.  }
  200. }
  201. SWORD get_svalue(BYTE k)
  202. // k>0 always
  203. // Takes k bits out of the BIT stream (wordval), and makes them a signed value
  204. {
  205. _asm {
  206.    xor ecx, ecx
  207.    mov cl,k
  208.    mov eax,[wordval]
  209.    shl eax,cl
  210.    shr eax, 16
  211.    dec cl
  212.    bt eax,ecx
  213.    jc end_macro
  214. signed_value:inc cl
  215.    mov ebx,[start_neg_pow2]
  216.    add ax,word ptr [ebx+ecx*2]
  217.  end_macro:
  218. }
  219. }
  220. #endif
  221. #ifdef __WATCOMC__
  222. WORD lookKbits(BYTE k);
  223. #pragma aux lookKbits=
  224.   "mov eax,[wordval]"
  225.   "mov cl, 16"
  226.   "sub cl, dl"
  227.   "shr eax, cl"
  228.    parm [dl] 
  229.    value [ax] 
  230.    modify [eax cl];
  231. WORD WORD_hi_lo(BYTE byte_high,BYTE BYTE_low);
  232. #pragma aux WORD_hi_lo=
  233.    parm [ah] [al]
  234.    value [ax] 
  235.    modify [ax];
  236. SWORD get_svalue(BYTE k);
  237. // k>0 always
  238. // Takes k bits out of the BIT stream (wordval), and makes them a signed value
  239. #pragma aux get_svalue=
  240. "xor ecx, ecx"
  241. "mov cl, al"
  242. "mov eax,[wordval]"
  243. "shl eax, cl"
  244. "shr eax, 16"
  245. "dec cl"
  246. "bt eax,ecx"
  247. "jc end_macro"
  248. "signed_value:inc cl"
  249. "mov ebx,[start_neg_pow2]"
  250. "add ax,word ptr [ebx+ecx*2]"
  251. "end_macro:"
  252. parm [al]
  253. modify [eax ebx ecx]
  254. value [ax];
  255. #endif
  256. void skipKbits(BYTE k)
  257. {
  258.  BYTE b_high,b_low;
  259.  d_k+=k;
  260.  if (d_k>=16) { d_k-=16;
  261. w1=w2;
  262. // Get the next word in w2
  263. BYTE_p(byte_pos);
  264. if (bp!=0xFF) b_high=bp;
  265. else {
  266.   if (buf[byte_pos]==0) byte_pos++; //skip 00
  267.   else byte_pos--; // stop byte_pos pe restart marker
  268.   b_high=0xFF;
  269. }
  270. BYTE_p(byte_pos);
  271. if (bp!=0xFF) b_low=bp;
  272. else {
  273.   if (buf[byte_pos]==0) byte_pos++; //skip 00
  274.   else byte_pos--; // stop byte_pos pe restart marker
  275.   b_low=0xFF;
  276. }
  277. w2=WORD_hi_lo(b_high,b_low);
  278.  }
  279.  wordval = ((DWORD)(w1)<<16) + w2;
  280.  wordval <<= d_k;
  281.  wordval >>= 16;
  282. }
  283. SWORD getKbits(BYTE k)
  284. {
  285.  SWORD signed_wordvalue;
  286.  signed_wordvalue=get_svalue(k);
  287.  skipKbits(k);
  288.  return signed_wordvalue;
  289. }
  290. void calculate_mask()
  291. {
  292.   BYTE k;
  293.   DWORD tmpdv;
  294.   for (k=0;k<=16;k++) { tmpdv=0x10000;mask[k]=(tmpdv>>k)-1;} //precalculated bit mask
  295. }
  296. void init_QT()
  297. {
  298.  BYTE i;
  299.  for (i=0;i<=3;i++) QT[i]=(float *)malloc(sizeof(float)*64);
  300. }
  301. void load_quant_table(float *quant_table)
  302. {
  303.  float scalefactor[8]={1.0f, 1.387039845f, 1.306562965f, 1.175875602f,
  304.    1.0f, 0.785694958f, 0.541196100f, 0.275899379f};
  305.  BYTE j,row,col;
  306. // Load quantization coefficients from JPG file, scale them for DCT and reorder
  307. // from zig-zag order
  308.  for (j=0;j<=63;j++) quant_table[j]=buf[byte_pos+zigzag[j]];
  309.  j=0;
  310.  for (row=0;row<=7;row++)
  311.    for (col=0;col<=7;col++) {
  312. quant_table[j]*=scalefactor[row]*scalefactor[col];
  313. j++;
  314.    }
  315.  byte_pos+=64;
  316. }
  317. void load_Huffman_table(Huffman_table *HT)
  318. {
  319.   BYTE k,j;
  320.   DWORD code;
  321.   for (j=1;j<=16;j++) {
  322. BYTE_p(byte_pos);
  323. HT->Length[j]=bp;
  324.   }
  325.   for (k=1;k<=16;k++)
  326. for (j=0;j<HT->Length[k];j++) {
  327. BYTE_p(byte_pos);
  328. HT->V[WORD_hi_lo(k,j)]=bp;
  329. }
  330.   code=0;
  331.   for (k=1;k<=16;k++) {
  332.  HT->minor_code[k] = (WORD)code;
  333.  for (j=1;j<=HT->Length[k];j++) code++;
  334.  HT->major_code[k]=(WORD)(code-1);
  335.  code*=2;
  336.  if (HT->Length[k]==0) {
  337. HT->minor_code[k]=0xFFFF;
  338. HT->major_code[k]=0;
  339.  }
  340.   }
  341. }
  342. void process_Huffman_data_unit(BYTE DC_nr, BYTE AC_nr,SWORD *previous_DC)
  343. {
  344. // Process one data unit. A data unit = 64 DCT coefficients
  345. // Data is decompressed by Huffman decoding, then the array is dezigzag-ed
  346. // The result is a 64 DCT coefficients array: DCT_coeff
  347.    BYTE nr,k,j,EOB_found;
  348.    register WORD tmp_Hcode;
  349.    BYTE size_val,count_0;
  350.    WORD *min_code,*maj_code; // min_code[k]=minimum code of length k, maj_code[k]=similar but maximum
  351.    WORD *max_val, *min_val;
  352.    BYTE *huff_values;
  353.    SWORD DCT_tcoeff[64];
  354.    BYTE byte_temp;
  355. // Start Huffman decoding
  356. // First the DC coefficient decoding
  357.    min_code=HTDC[DC_nr].minor_code;
  358.    maj_code=HTDC[DC_nr].major_code;
  359.    huff_values=HTDC[DC_nr].V;
  360.    for (nr = 0; nr < 64 ; nr++) DCT_tcoeff[nr] = 0; //Initialize DCT_tcoeff
  361.    nr=0;// DC coefficient
  362.    min_val = &min_code[1]; max_val = &maj_code[1];
  363.    for (k=1;k<=16;k++) {
  364.  tmp_Hcode=lookKbits(k);
  365. //    max_val = &maj_code[k]; min_val = &min_code[k];
  366.  if ( (tmp_Hcode<=*max_val)&&(tmp_Hcode>=*min_val) ) { //Found a valid Huffman code
  367. skipKbits(k);
  368. size_val=huff_values[WORD_hi_lo(k,(BYTE)(tmp_Hcode-*min_val))];
  369. if (size_val==0) DCT_tcoeff[0]=*previous_DC;
  370. else {
  371. DCT_tcoeff[0]=*previous_DC+getKbits(size_val);
  372. *previous_DC=DCT_tcoeff[0];
  373. }
  374. break;
  375.  }
  376.  min_val++; max_val++;
  377.    }
  378. // Second, AC coefficient decoding
  379.    min_code=HTAC[AC_nr].minor_code;
  380.    maj_code=HTAC[AC_nr].major_code;
  381.    huff_values=HTAC[AC_nr].V;
  382.    nr=1; // AC coefficient
  383.    EOB_found=0;
  384.    while ( (nr<=63)&&(!EOB_found) )
  385. {
  386.  max_val = &maj_code[1]; min_val =&min_code[1];
  387.  for (k=1;k<=16;k++)
  388.  {
  389.    tmp_Hcode=lookKbits(k);
  390. //    max_val = &maj_code[k]; &min_val = min_code[k];
  391.    if ( (tmp_Hcode<=*max_val)&&(tmp_Hcode>=*min_val) )
  392.    {
  393. skipKbits(k);
  394. byte_temp=huff_values[WORD_hi_lo(k,(BYTE)(tmp_Hcode-*min_val))];
  395. size_val=byte_temp&0xF;
  396. count_0=byte_temp>>4;
  397. if (size_val==0) {if (count_0==0) EOB_found=1;
  398.   else if (count_0==0xF) nr+=16; //skip 16 zeroes
  399. }
  400. else
  401. {
  402.  nr+=count_0; //skip count_0 zeroes
  403.  DCT_tcoeff[nr++]=getKbits(size_val);
  404. }
  405. break;
  406.    }
  407.    min_val++; max_val++;
  408.  }
  409.  if (k>16) nr++;  // This should not occur
  410. }
  411.   for (j=0;j<=63;j++) DCT_coeff[j]=DCT_tcoeff[zigzag[j]]; // Et, voila ... DCT_coeff
  412. }
  413. void IDCT_transform(SWORD *incoeff,BYTE *outcoeff,BYTE Q_nr)
  414. // Fast float IDCT transform
  415. {
  416.  BYTE x;
  417.  SBYTE y;
  418.  SWORD *inptr;
  419.  BYTE *outptr;
  420.  float workspace[64];
  421.  float *wsptr;//Workspace pointer
  422.  float *quantptr; // Quantization table pointer
  423.  float dcval;
  424.  float tmp0,tmp1,tmp2,tmp3,tmp4,tmp5,tmp6,tmp7;
  425.  float tmp10,tmp11,tmp12,tmp13;
  426.  float z5,z10,z11,z12,z13;
  427.  BYTE *range_limit=rlimit_table+128;
  428.  // Pass 1: process COLUMNS from input and store into work array.
  429.  wsptr=workspace;
  430.  inptr=incoeff;
  431.  quantptr=QT[Q_nr];
  432.  for (y=0;y<=7;y++)
  433.   {
  434.    if( (inptr[8]|inptr[16]|inptr[24]|inptr[32]|inptr[40]|inptr[48]|inptr[56])==0)
  435. {
  436.  // AC terms all zero (in a column)
  437.  dcval=inptr[0]*quantptr[0];
  438.  wsptr[0]  = dcval;
  439.  wsptr[8]  = dcval;
  440.  wsptr[16] = dcval;
  441.  wsptr[24] = dcval;
  442.  wsptr[32] = dcval;
  443.  wsptr[40] = dcval;
  444.  wsptr[48] = dcval;
  445.  wsptr[56] = dcval;
  446.  inptr++;quantptr++;wsptr++;//advance pointers to next column
  447.  continue ;
  448. }
  449.   //Even part
  450. tmp0 = inptr[0] *quantptr[0];
  451. tmp1 = inptr[16]*quantptr[16];
  452. tmp2 = inptr[32]*quantptr[32];
  453. tmp3 = inptr[48]*quantptr[48];
  454. tmp10 = tmp0 + tmp2;// phase 3
  455. tmp11 = tmp0 - tmp2;
  456. tmp13 = tmp1 + tmp3;// phases 5-3
  457. tmp12 = (tmp1 - tmp3) * 1.414213562f - tmp13; // 2*c4
  458. tmp0 = tmp10 + tmp13;// phase 2
  459. tmp3 = tmp10 - tmp13;
  460. tmp1 = tmp11 + tmp12;
  461. tmp2 = tmp11 - tmp12;
  462. // Odd part
  463. tmp4 = inptr[8] *quantptr[8];
  464. tmp5 = inptr[24]*quantptr[24];
  465. tmp6 = inptr[40]*quantptr[40];
  466. tmp7 = inptr[56]*quantptr[56];
  467. z13 = tmp6 + tmp5;// phase 6
  468. z10 = tmp6 - tmp5;
  469. z11 = tmp4 + tmp7;
  470. z12 = tmp4 - tmp7;
  471. tmp7 = z11 + z13;// phase 5
  472. tmp11= (z11 - z13) * 1.414213562f; // 2*c4
  473. z5 = (z10 + z12) * 1.847759065f; // 2*c2
  474. tmp10 = 1.082392200f * z12 - z5; // 2*(c2-c6)
  475. tmp12 = -2.613125930f * z10 + z5;// -2*(c2+c6)
  476. tmp6 = tmp12 - tmp7;// phase 2
  477. tmp5 = tmp11 - tmp6;
  478. tmp4 = tmp10 + tmp5;
  479. wsptr[0]  = tmp0 + tmp7;
  480. wsptr[56] = tmp0 - tmp7;
  481. wsptr[8]  = tmp1 + tmp6;
  482. wsptr[48] = tmp1 - tmp6;
  483. wsptr[16] = tmp2 + tmp5;
  484. wsptr[40] = tmp2 - tmp5;
  485. wsptr[32] = tmp3 + tmp4;
  486. wsptr[24] = tmp3 - tmp4;
  487. inptr++;
  488. quantptr++;
  489. wsptr++;//advance pointers to the next column
  490.   }
  491. //  Pass 2: process ROWS from work array, store into output array.
  492. // Note that we must descale the results by a factor of 8 = 2^3
  493.  outptr=outcoeff;
  494.  wsptr=workspace;
  495.  for (x=0;x<=7;x++)
  496.   {
  497.    // Even part
  498. tmp10 = wsptr[0] + wsptr[4];
  499. tmp11 = wsptr[0] - wsptr[4];
  500. tmp13 = wsptr[2] + wsptr[6];
  501. tmp12 =(wsptr[2] - wsptr[6]) * 1.414213562f - tmp13;
  502. tmp0 = tmp10 + tmp13;
  503. tmp3 = tmp10 - tmp13;
  504. tmp1 = tmp11 + tmp12;
  505. tmp2 = tmp11 - tmp12;
  506.    // Odd part
  507. z13 = wsptr[5] + wsptr[3];
  508. z10 = wsptr[5] - wsptr[3];
  509. z11 = wsptr[1] + wsptr[7];
  510. z12 = wsptr[1] - wsptr[7];
  511. tmp7 = z11 + z13;
  512. tmp11= (z11 - z13) * 1.414213562f;
  513. z5 = (z10 + z12) * 1.847759065f; // 2*c2
  514. tmp10 = 1.082392200f * z12 - z5;  // 2*(c2-c6)
  515. tmp12 = -2.613125930f * z10 + z5; // -2*(c2+c6)
  516. tmp6 = tmp12 - tmp7;
  517. tmp5 = tmp11 - tmp6;
  518. tmp4 = tmp10 + tmp5;
  519.  // Final output stage: scale down by a factor of 8
  520. outptr[0] = range_limit[(DESCALE((int) (tmp0 + tmp7), 3)) & 1023L];
  521. outptr[7] = range_limit[(DESCALE((int) (tmp0 - tmp7), 3)) & 1023L];
  522. outptr[1] = range_limit[(DESCALE((int) (tmp1 + tmp6), 3)) & 1023L];
  523. outptr[6] = range_limit[(DESCALE((int) (tmp1 - tmp6), 3)) & 1023L];
  524. outptr[2] = range_limit[(DESCALE((int) (tmp2 + tmp5), 3)) & 1023L];
  525. outptr[5] = range_limit[(DESCALE((int) (tmp2 - tmp5), 3)) & 1023L];
  526. outptr[4] = range_limit[(DESCALE((int) (tmp3 + tmp4), 3)) & 1023L];
  527. outptr[3] = range_limit[(DESCALE((int) (tmp3 - tmp4), 3)) & 1023L];
  528. wsptr+=8;//advance pointer to the next row
  529. outptr+=8;
  530.   }
  531. }
  532. void precalculate_Cr_Cb_tables()
  533. {
  534.  WORD k;
  535.  WORD Cr_v,Cb_v;
  536.  for (k=0;k<=255;k++) Cr_tab[k]=(SWORD)((k-128.0)*1.402);
  537.  for (k=0;k<=255;k++) Cb_tab[k]=(SWORD)((k-128.0)*1.772);
  538.  for (Cr_v=0;Cr_v<=255;Cr_v++)
  539.   for (Cb_v=0;Cb_v<=255;Cb_v++)
  540.    Cr_Cb_green_tab[((WORD)(Cr_v)<<8)+Cb_v]=(int)(-0.34414*(Cb_v-128.0)-0.71414*(Cr_v-128.0));
  541. }
  542. void convert_8x8_YCbCr_to_RGB(BYTE *Y, BYTE *Cb, BYTE *Cr, DWORD im_loc, DWORD X_image_bytes, BYTE *im_buffer)
  543. // Functia (ca optimizare) poate fi apelata si fara parametrii Y,Cb,Cr
  544. // Stim ca va fi apelata doar in cazul 1x1
  545. {
  546.   DWORD x,y;
  547.   BYTE im_nr;
  548.   BYTE *Y_val = Y, *Cb_val = Cb, *Cr_val = Cr;
  549.   BYTE *ibuffer = im_buffer + im_loc;
  550.   for (y=0;y<8;y++)
  551.    {
  552. im_nr=0;
  553. for (x=0;x<8;x++)
  554.   {
  555.    ibuffer[im_nr++] = rlimit_table[*Y_val + Cb_tab[*Cb_val]]; //B
  556.    ibuffer[im_nr++] = rlimit_table[*Y_val + Cr_Cb_green_tab[WORD_hi_lo(*Cr_val,*Cb_val)]]; //G
  557.    ibuffer[im_nr++] = rlimit_table[*Y_val + Cr_tab[*Cr_val]]; // R
  558. /*
  559. // Monochrome display
  560.    im_buffer[im_nr++] = *Y_val;
  561.    im_buffer[im_nr++] = *Y_val;
  562.    im_buffer[im_nr++] = *Y_val;
  563. */
  564.    Y_val++; Cb_val++; Cr_val++; im_nr++;
  565.   }
  566. ibuffer+=X_image_bytes;
  567.    }
  568. }
  569. void convert_8x8_YCbCr_to_RGB_tab(BYTE *Y, BYTE *Cb, BYTE *Cr, BYTE *tab, DWORD im_loc, DWORD X_image_bytes, BYTE *im_buffer)
  570. // Functia (ca optimizare) poate fi apelata si fara parametrii Cb,Cr
  571. {
  572.   DWORD x,y;
  573.   BYTE nr, im_nr;
  574.   BYTE Y_val,Cb_val,Cr_val;
  575.   BYTE *ibuffer = im_buffer + im_loc;
  576.   nr=0;
  577.   for (y=0;y<8;y++)
  578.    {
  579. im_nr=0;
  580. for (x=0;x<8;x++)
  581.   {
  582.    Y_val=Y[nr];
  583.    Cb_val=Cb[tab[nr]]; Cr_val=Cr[tab[nr]]; // reindexare folosind tabelul
  584.    // de supraesantionare precalculat
  585.    ibuffer[im_nr++] = rlimit_table[Y_val + Cb_tab[Cb_val]]; //B
  586.    ibuffer[im_nr++] = rlimit_table[Y_val + Cr_Cb_green_tab[WORD_hi_lo(Cr_val,Cb_val)]]; //G
  587.    ibuffer[im_nr++] = rlimit_table[Y_val + Cr_tab[Cr_val]]; // R
  588.    nr++; im_nr++;
  589.   }
  590. ibuffer+=X_image_bytes;
  591.    }
  592. }
  593. void calculate_tabs()
  594. {
  595.  BYTE tab_temp[256];
  596.  BYTE x,y;
  597.  // Tabelul de supraesantionare 16x16
  598.  for (y=0;y<16;y++)
  599.  for (x=0;x<16;x++)
  600.    tab_temp[y*16+x] = (y/YV)* 8 + x/YH;
  601.  // Din el derivam tabelele corespunzatoare celor 4 blocuri de 8x8 pixeli
  602.  for (y=0;y<8;y++)
  603. {
  604.  for (x=0;x<8;x++)
  605.   tab_1[y*8+x]=tab_temp[y*16+x];
  606.  for (x=8;x<16;x++)
  607.   tab_2[y*8+(x-8)]=tab_temp[y*16+x];
  608. }
  609.  for (y=8;y<16;y++)
  610. {
  611.  for (x=0;x<8;x++)
  612.   tab_3[(y-8)*8+x]=tab_temp[y*16+x];
  613.  for (x=8;x<16;x++)
  614.   tab_4[(y-8)*8+(x-8)]=tab_temp[y*16+x];
  615. }
  616. }
  617. int init_JPG_decoding()
  618. {
  619.  byte_pos=0;
  620.  init_QT();
  621.  calculate_mask();
  622.  prepare_range_limit_table();
  623.  precalculate_Cr_Cb_tables();
  624.  return 1; //for future error check
  625. }
  626. DWORD filesize(FILE *fp)
  627. {
  628.  DWORD pos;
  629.  DWORD pos_cur;
  630.  pos_cur=ftell(fp);
  631.  fseek(fp,0,SEEK_END);
  632.  pos=ftell(fp);
  633.  fseek(fp,pos_cur,SEEK_SET);
  634.  return pos;
  635. }
  636. int load_JPEG_header(FILE *fp, DWORD *X_image, DWORD *Y_image)
  637. {
  638.  DWORD length_of_file;
  639.  BYTE vers,units;
  640.  WORD Xdensity,Ydensity,Xthumbnail,Ythumbnail;
  641.  WORD length;
  642.  float *qtable;
  643.  DWORD old_byte_pos;
  644.  Huffman_table *htable;
  645.  DWORD j;
  646.  BYTE precision,comp_id,nr_components;
  647.  BYTE QT_info,HT_info;
  648.  BYTE SOS_found,SOF_found;
  649.  length_of_file=filesize(fp);
  650.  buf=(BYTE *)malloc(length_of_file+4);
  651.  if (buf==NULL) exit_func("Not enough memory for loading file");
  652.  fread(buf,length_of_file,1,fp);
  653.  if ((buf[0]!=0xFF)||(buf[1]!=SOI)) exit_func("Not a JPG file ?n");
  654.  if ((buf[2]!=0xFF)||(buf[3]!=APP0)) exit_func("Invalid JPG file.");
  655.  if ( (buf[6]!='J')||(buf[7]!='F')||(buf[8]!='I')||(buf[9]!='F')||
  656.   (buf[10]!=0) ) exit_func("Invalid JPG file.");
  657.  init_JPG_decoding();
  658.  byte_pos=11;
  659.  BYTE_p(byte_pos);vers=bp;
  660.  if (vers!=1) exit_func("JFIF version not supported");
  661.  BYTE_p(byte_pos); // vers_lo=bp;
  662.  BYTE_p(byte_pos);  units=bp;
  663.  if (units!=0) //exit_func("JPG format not supported");
  664. ;// printf("units = %dn", units);
  665.  WORD_p(byte_pos); Xdensity=wp; WORD_p(byte_pos); Ydensity=wp;
  666.  if ((Xdensity!=1)||(Ydensity!=1)) //exit_func("JPG format not supported");
  667.  ;  //{printf("X density = %dn",Xdensity); printf("Y density = %dn",Ydensity);}
  668.  BYTE_p(byte_pos);Xthumbnail=bp;BYTE_p(byte_pos);Ythumbnail=bp;
  669.  if ((Xthumbnail!=0)||(Ythumbnail!=0))
  670. exit_func(" Cannot process JFIF thumbnailed filesn");
  671.  // Start decoding process
  672.  SOS_found=0; SOF_found=0; Restart_markers=0;
  673.  while ((byte_pos<length_of_file)&&!SOS_found)
  674.  {
  675.   BYTE_p(byte_pos);
  676.   if (bp!=0xFF) continue;
  677.   // A marker was found
  678.   BYTE_p(byte_pos);
  679.   switch(bp)
  680.   {
  681.    case DQT: WORD_p(byte_pos); length=wp; // length of the DQT marker
  682.  for (j=0;j<wp-2;)
  683. {
  684.  old_byte_pos=byte_pos;
  685.  BYTE_p(byte_pos); QT_info=bp;
  686.  if ((QT_info>>4)!=0)
  687.  exit_func("16 bit quantization table not supported");
  688.  qtable=QT[QT_info&0xF];
  689.  load_quant_table(qtable);
  690.  j+=byte_pos-old_byte_pos;
  691. }
  692.  break;
  693.    case DHT: WORD_p(byte_pos); length=wp;
  694.  for (j=0;j<wp-2;)
  695. {
  696.  old_byte_pos=byte_pos;
  697.  BYTE_p(byte_pos); HT_info=bp;
  698.  if ((HT_info&0x10)!=0) htable=&HTAC[HT_info&0xF];
  699.  else htable=&HTDC[HT_info&0xF];
  700.  load_Huffman_table(htable);
  701.  j+=byte_pos-old_byte_pos;
  702. }
  703.  break;
  704.    case COM: WORD_p(byte_pos); length=wp;
  705.  byte_pos+=wp-2;
  706.  break;
  707.    case DRI: Restart_markers=1;
  708.  WORD_p(byte_pos); length=wp; //should be = 4
  709.  WORD_p(byte_pos);  MCU_restart=wp;
  710.  if (MCU_restart==0) Restart_markers=0;
  711.  break;
  712.    case SOF: WORD_p(byte_pos); length=wp; //should be = 8+3*3=17
  713.  BYTE_p(byte_pos); precision=bp;
  714.  if (precision!=8) exit_func("Only 8 bit precision supported");
  715.  WORD_p(byte_pos); *Y_image=wp; WORD_p(byte_pos); *X_image=wp;
  716.  BYTE_p(byte_pos); nr_components=bp;
  717.  if (nr_components!=3) exit_func("Only truecolor JPGS supported");
  718.  for (j=1;j<=3;j++)
  719. {
  720.  BYTE_p(byte_pos); comp_id=bp;
  721.  if ((comp_id==0)||(comp_id>3)) exit_func("Only YCbCr format supported");
  722.  switch (comp_id)
  723. {
  724.  case 1: // Y
  725. BYTE_p(byte_pos); YH=bp>>4;YV=bp&0xF;
  726. BYTE_p(byte_pos); YQ_nr=bp;
  727. break;
  728.  case 2: // Cb
  729. BYTE_p(byte_pos); CbH=bp>>4;CbV=bp&0xF;
  730. BYTE_p(byte_pos); CbQ_nr=bp;
  731. break;
  732.  case 3: // Cr
  733. BYTE_p(byte_pos); CrH=bp>>4;CrV=bp&0xF;
  734. BYTE_p(byte_pos); CrQ_nr=bp;
  735. break;
  736. }
  737. }
  738.  SOF_found=1;
  739.  break;
  740.    case SOS: WORD_p(byte_pos); length=wp; //should be = 6+3*2=12
  741.  BYTE_p(byte_pos); nr_components=bp;
  742.  if (nr_components!=3) exit_func("Invalid SOS marker");
  743.  for (j=1;j<=3;j++)
  744.    {
  745. BYTE_p(byte_pos); comp_id=bp;
  746. if ((comp_id==0)||(comp_id>3)) exit_func("Only YCbCr format supported");
  747. switch (comp_id)
  748. {
  749.  case 1: // Y
  750. BYTE_p(byte_pos); YDC_nr=bp>>4;YAC_nr=bp&0xF;
  751. break;
  752.  case 2: // Cb
  753. BYTE_p(byte_pos); CbDC_nr=bp>>4;CbAC_nr=bp&0xF;
  754. break;
  755.  case 3: // Cr
  756. BYTE_p(byte_pos); CrDC_nr=bp>>4;CrAC_nr=bp&0xF;
  757. break;
  758. }
  759.    }
  760.  BYTE_p(byte_pos); BYTE_p(byte_pos); BYTE_p(byte_pos); // Skip 3 bytes
  761.  SOS_found=1;
  762.  break;
  763.    case 0xFF:
  764.  break; // do nothing for 0xFFFF, sequence of consecutive 0xFF are for
  765. // filling purposes and should be ignored
  766.    default:  WORD_p(byte_pos); length=wp;
  767.  byte_pos+=wp-2; //skip unknown marker
  768.  break;
  769.   }
  770.  }
  771.  if (!SOS_found) exit_func("Invalid JPG file. No SOS marker found.");
  772.  if (!SOF_found) exit_func("Progressive JPEGs not supported");
  773.  if ((CbH>YH)||(CrH>YH)) exit_func("Vertical sampling factor for Y should be >= sampling factor for Cb,Cr");
  774.  if ((CbV>YV)||(CrV>YV)) exit_func("Horizontal sampling factor for Y should be >= sampling factor for Cb,Cr");
  775.  if ((CbH>=2)||(CbV>=2)) exit_func("Cb sampling factors should be = 1");
  776.  if ((CrV>=2)||(CrV>=2)) exit_func("Cr sampling factors should be = 1");
  777. // Restricting sampling factors for Y,Cb,Cr should give us 4 possible cases for sampling factors
  778. // YHxYV,CbHxCbV,CrHxCrV: 2x2,1x1,1x1;  1x2,1x1,1x1; 2x1,1x1,1x1;
  779. // and 1x1,1x1,1x1 = no upsampling needed
  780.  Hmax=YH,Vmax=YV;
  781.  if ( *X_image%(Hmax*8)==0) X_round=*X_image; // X_round = Multiple of Hmax*8
  782.  else X_round=(*X_image/(Hmax*8)+1)*(Hmax*8);
  783.  if ( *Y_image%(Vmax*8)==0) Y_round=*Y_image; // Y_round = Multiple of Vmax*8
  784.  else Y_round=(*Y_image/(Vmax*8)+1)*(Vmax*8);
  785.  im_buffer=(BYTE *)malloc(X_round*Y_round*4);
  786.  if (im_buffer==NULL) exit_func("Not enough memory for storing the JPEG image");
  787.  return 1;
  788. }
  789. void resync()
  790. // byte_pos  = pozitionat pe restart marker
  791. {
  792.  byte_pos+=2;
  793.  BYTE_p(byte_pos);
  794.  if (bp==0xFF) byte_pos++; // skip 00
  795.  w1=WORD_hi_lo(bp, 0);
  796.  BYTE_p(byte_pos);
  797.  if (bp==0xFF) byte_pos++; // skip 00
  798.  w1+=bp;
  799.  BYTE_p(byte_pos);
  800.  if (bp==0xFF) byte_pos++; // skip 00
  801.  w2=WORD_hi_lo(bp, 0);
  802.  BYTE_p(byte_pos);
  803.  if (bp==0xFF) byte_pos++; // skip 00
  804.  w2+=bp;
  805.  wordval=w1; d_k=0; // Reinit bitstream decoding
  806.  DCY=0; DCCb=0; DCCr=0; // Init DC coefficients
  807. }
  808. void decode_MCU_1x1(DWORD im_loc)
  809. {
  810.  // Y
  811.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  812.  IDCT_transform(DCT_coeff,Y,YQ_nr);
  813.  // Cb
  814.  process_Huffman_data_unit(CbDC_nr,CbAC_nr,&DCCb);
  815.  IDCT_transform(DCT_coeff,Cb,CbQ_nr);
  816.  // Cr
  817.  process_Huffman_data_unit(CrDC_nr,CrAC_nr,&DCCr);
  818.  IDCT_transform(DCT_coeff,Cr,CrQ_nr);
  819.  convert_8x8_YCbCr_to_RGB(Y,Cb,Cr,im_loc,X_image_bytes,im_buffer);
  820. }
  821. void decode_MCU_2x1(DWORD im_loc)
  822. {
  823.  // Y
  824.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  825.  IDCT_transform(DCT_coeff,Y_1,YQ_nr);
  826.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  827.  IDCT_transform(DCT_coeff,Y_2,YQ_nr);
  828.  // Cb
  829.  process_Huffman_data_unit(CbDC_nr,CbAC_nr,&DCCb);
  830.  IDCT_transform(DCT_coeff,Cb,CbQ_nr);
  831.  // Cr
  832.  process_Huffman_data_unit(CrDC_nr,CrAC_nr,&DCCr);
  833.  IDCT_transform(DCT_coeff,Cr,CrQ_nr);
  834.  convert_8x8_YCbCr_to_RGB_tab(Y_1,Cb,Cr,tab_1,im_loc,X_image_bytes,im_buffer);
  835.  convert_8x8_YCbCr_to_RGB_tab(Y_2,Cb,Cr,tab_2,im_loc+32,X_image_bytes,im_buffer);
  836. }
  837. void decode_MCU_2x2(DWORD im_loc)
  838. {
  839.  // Y
  840.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  841.  IDCT_transform(DCT_coeff,Y_1,YQ_nr);
  842.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  843.  IDCT_transform(DCT_coeff,Y_2,YQ_nr);
  844.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  845.  IDCT_transform(DCT_coeff,Y_3,YQ_nr);
  846.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  847.  IDCT_transform(DCT_coeff,Y_4,YQ_nr);
  848.  // Cb
  849.  process_Huffman_data_unit(CbDC_nr,CbAC_nr,&DCCb);
  850.  IDCT_transform(DCT_coeff,Cb,CbQ_nr);
  851.  // Cr
  852.  process_Huffman_data_unit(CrDC_nr,CrAC_nr,&DCCr);
  853.  IDCT_transform(DCT_coeff,Cr,CrQ_nr);
  854.  convert_8x8_YCbCr_to_RGB_tab(Y_1,Cb,Cr,tab_1,im_loc,X_image_bytes,im_buffer);
  855.  convert_8x8_YCbCr_to_RGB_tab(Y_2,Cb,Cr,tab_2,im_loc+32,X_image_bytes,im_buffer);
  856.  convert_8x8_YCbCr_to_RGB_tab(Y_3,Cb,Cr,tab_3,im_loc+y_inc_value,X_image_bytes,im_buffer);
  857.  convert_8x8_YCbCr_to_RGB_tab(Y_4,Cb,Cr,tab_4,im_loc+y_inc_value+32,X_image_bytes,im_buffer);
  858. }
  859. void decode_MCU_1x2(DWORD im_loc)
  860. {
  861.  // Y
  862.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  863.  IDCT_transform(DCT_coeff,Y_1,YQ_nr);
  864.  process_Huffman_data_unit(YDC_nr,YAC_nr,&DCY);
  865.  IDCT_transform(DCT_coeff,Y_2,YQ_nr);
  866.  // Cb
  867.  process_Huffman_data_unit(CbDC_nr,CbAC_nr,&DCCb);
  868.  IDCT_transform(DCT_coeff,Cb,CbQ_nr);
  869.  // Cr
  870.  process_Huffman_data_unit(CrDC_nr,CrAC_nr,&DCCr);
  871.  IDCT_transform(DCT_coeff,Cr,CrQ_nr);
  872.  convert_8x8_YCbCr_to_RGB_tab(Y_1,Cb,Cr,tab_1,im_loc,X_image_bytes,im_buffer);
  873.  convert_8x8_YCbCr_to_RGB_tab(Y_2,Cb,Cr,tab_3,im_loc+y_inc_value,X_image_bytes,im_buffer);
  874. }
  875. void decode_JPEG_image()
  876. {
  877.  decode_MCU_func decode_MCU;
  878.  WORD x_mcu_cnt,y_mcu_cnt;
  879.  DWORD nr_mcu;
  880.  WORD X_MCU_nr,Y_MCU_nr; // Nr de MCU-uri
  881.  DWORD MCU_dim_x; //dimensiunea in bufferul imagine a unui MCU pe axa x
  882.  DWORD im_loc_inc; // = 7 * X_round * 4 sau 15*X_round*4;
  883.  DWORD im_loc; //locatia in bufferul imagine
  884.  byte_pos-=2;
  885.  resync();
  886.  y_inc_value = 32*X_round;
  887.  calculate_tabs(); // Calcul tabele de supraesantionare, tinand cont de YH si YV
  888.  if ((YH==1)&&(YV==1)) decode_MCU=decode_MCU_1x1;
  889.  else {
  890.    if (YH==2)
  891.    {
  892. if (YV==2) decode_MCU=decode_MCU_2x2;
  893. else decode_MCU=decode_MCU_2x1;
  894.    }
  895.    else decode_MCU=decode_MCU_1x2;
  896.  }
  897.  MCU_dim_x=Hmax*8*4;
  898.  Y_MCU_nr=Y_round/(Vmax*8); // nr of MCUs on Y axis
  899.  X_MCU_nr=X_round/(Hmax*8); // nr of MCUs on X axis
  900.  X_image_bytes=X_round*4; im_loc_inc = (Vmax*8-1) * X_image_bytes;
  901.  nr_mcu=0; im_loc=0; // memory location of the current MCU
  902.  for (y_mcu_cnt=0;y_mcu_cnt<Y_MCU_nr;y_mcu_cnt++)
  903.  {
  904.   for (x_mcu_cnt=0;x_mcu_cnt<X_MCU_nr;x_mcu_cnt++)
  905.    {
  906. decode_MCU(im_loc);
  907. if ((Restart_markers)&&((nr_mcu+1)%MCU_restart==0)) resync();
  908. nr_mcu++;
  909. im_loc+=MCU_dim_x;
  910.    }
  911.   im_loc+=im_loc_inc;
  912.  }
  913. }
  914. int get_JPEG_buffer(WORD X_image,WORD Y_image, BYTE **address_dest_buffer)
  915. {
  916.  WORD y;
  917.  DWORD dest_loc=0;
  918.  BYTE *src_buffer=im_buffer;
  919.  BYTE *dest_buffer_start, *dest_buffer;
  920.  DWORD src_bytes_per_line=X_round*4;
  921.  DWORD dest_bytes_per_line=X_image*4;
  922.  if ((X_round==X_image)&&(Y_round==Y_image))
  923. *address_dest_buffer=im_buffer;
  924.  else
  925.  {
  926.   dest_buffer_start = (BYTE *)malloc(X_image*Y_image*4);
  927.   if (dest_buffer_start==NULL) exit_func("Not enough memory for storing the JPEG image");
  928.   dest_buffer = dest_buffer_start;
  929.   for (y=0;y<Y_image;y++) {
  930.    memcpy(dest_buffer,src_buffer,dest_bytes_per_line);
  931.    src_buffer+=src_bytes_per_line;
  932.    dest_buffer+=dest_bytes_per_line;
  933.   }
  934.  *address_dest_buffer=dest_buffer_start;
  935.  free(im_buffer);
  936.  }
  937. // release the buffer which contains the JPG file
  938.  free(buf);
  939.  return 1;
  940. }
  941. void main(int argc, char *argv[])
  942. {
  943.  FILE *fp;
  944.  DWORD X_image, Y_image;
  945.  BYTE *our_image_buffer;
  946.  clock_t start_time, finish_time;
  947.  float duration;
  948.  if (argc<=1) fp=fopen(FileName,"rb");
  949.   else fp=fopen(argv[1],"rb");
  950.  if (fp==NULL) exitmessage("File not found ?");
  951.  if (!load_JPEG_header(fp,&X_image,&Y_image)) {exitmessage(error_string);return;}
  952.  fclose(fp);
  953.  printf(" X_image = %dn",X_image);
  954.  printf(" Y_image = %dn",Y_image);
  955. /*
  956.  printf("Sampling factors: n");
  957.  printf("Y  : H=%d,V=%dn", YH,YV);
  958.  printf("Cb : H=%d,V=%dn", CbH,CbV);
  959.  printf("Cr : H=%d,V=%dn", CrH,CrV);
  960.  printf("Restart markers  = %dn", Restart_markers);
  961.  printf("MCU restart = %dn", MCU_restart);
  962.  getch();
  963. */
  964.  printf("Decoding JPEG image...n");
  965.  // main decoder
  966.  start_time = clock();
  967.  decode_JPEG_image();
  968.  printf("Decoding finished.n");
  969.  
  970.  finish_time = clock();
  971.  duration = (double)(finish_time - start_time) / CLK_TCK;
  972.  printf( "Time elapsed: %2.1f secondsn", duration );
  973.  if (!get_JPEG_buffer(X_image,Y_image,&our_image_buffer)) {exitmessage(error_string);return;}
  974.  write_buf_to_BMP(our_image_buffer,X_image,Y_image, "image.bmp");
  975.  getch();
  976. }