简单的JPEG解码程序

转载自:

http://blog.csdn.net/lpt19832003/article/details/1713708

// JPGFile.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"

// simplejpeg.cpp : Defines the entry point for the console application.
//
//header file
//#include "bmp.h"


//*************************************************************************************
//#include "jpeg.h"
#include <stdio.h>
#include <stdlib.h>

#pragma pack(1)
#define M_SOF0  0xc0
#define M_DHT   0xc4
#define M_EOI   0xd9
#define M_SOS   0xda
#define M_DQT   0xdb
#define M_DRI   0xdd
#define M_APP0  0xe0

static int Zig_Zag[8][8]={{0,1,5,6,14,15,27,28},
        {2,4,7,13,16,26,29,42},
        {3,8,12,17,25,30,41,43},
        {9,11,18,24,37,40,44,53},
        {10,19,23,32,39,45,52,54},
        {20,22,33,38,46,51,55,60},
        {21,34,37,47,50,56,59,61},
        {35,36,48,49,57,58,62,63}
       };

#define W1 2841 /* 2048*sqrt(2)*cos(1*pi/16) */
#define W2 2676 /* 2048*sqrt(2)*cos(2*pi/16) */
#define W3 2408 /* 2048*sqrt(2)*cos(3*pi/16) */
#define W5 1609 /* 2048*sqrt(2)*cos(5*pi/16) */
#define W6 1108 /* 2048*sqrt(2)*cos(6*pi/16) */
#define W7 565  /* 2048*sqrt(2)*cos(7*pi/16) */


//*************************************************************************************
typedef char CHAR;
typedef short SHORT;
typedef long LONG;

 

typedef unsigned long       DWORD;
typedef int                 BOOL;
typedef unsigned char       BYTE;
typedef unsigned short      WORD;

 

typedef int HFILE;
typedef CHAR *LPSTR, *PSTR;

#define FALSE 0
#define TRUE 1


typedef struct tagBITMAPINFOHEADER{
        DWORD      biSize;
        LONG       biWidth;
        LONG       biHeight;
        WORD       biPlanes;
        WORD       biBitCount;
        DWORD      biCompression;
        DWORD      biSizeImage;
        LONG       biXPelsPerMeter;
        LONG       biYPelsPerMeter;
        DWORD      biClrUsed;
        DWORD      biClrImportant;
} BITMAPINFOHEADER, * LPBITMAPINFOHEADER,*PBITMAPINFOHEADER;

typedef struct tagBITMAPFILEHEADER {
        WORD    bfType;
        DWORD   bfSize;
        WORD    bfReserved1;
        WORD    bfReserved2;
        DWORD   bfOffBits;
}  BITMAPFILEHEADER,  * LPBITMAPFILEHEADER,*PBITMAPFILEHEADER;

/* constants for the biCompression field */
#define BI_RGB        0L
#define BI_RLE8       1L
#define BI_RLE4       2L
#define BI_BITFIELDS  3L


typedef struct tagRGBQUAD {
        BYTE    rgbBlue;
        BYTE    rgbGreen;
        BYTE    rgbRed;
        BYTE    rgbReserved;
} RGBQUAD;
typedef RGBQUAD * LPRGBQUAD;


#define MAKEWORD(a, b)      ((WORD)(((BYTE)(a)) | ((WORD)((BYTE)(b))) << 8))
#define MAKELONG(a, b)      ((LONG)(((WORD)(a)) | ((DWORD)((WORD)(b))) << 16))
#define LOWORD(l)           ((WORD)(l))
#define HIWORD(l)           ((WORD)(((DWORD)(l) >> 16) & 0xFFFF))
#define LOBYTE(w)           ((BYTE)(w))
#define HIBYTE(w)           ((BYTE)(((WORD)(w) >> 8) & 0xFF))

//---yk--- add

 


#include "memory.h"
#include "math.h"
#include "stdio.h"
//macro definition
#define WIDTHBYTES(i)    ((i+31)/32*4)//??????????
#define PI 3.1415926535
//define return value of function
#define FUNC_OK 0
#define FUNC_MEMORY_ERROR 1
#define FUNC_FILE_ERROR 2
#define FUNC_FORMAT_ERROR 3


//////////////////////////////////////////////////
//Jpeg functions
BOOL LoadJpegFile(char *BmpFileName);
void showerror(int funcret);
int  InitTag();
void InitTable();
int  Decode();
int  DecodeMCUBlock();
int  HufBlock(BYTE dchufindex,BYTE achufindex);
int  DecodeElement();
void IQtIZzMCUComponent(short flag);
void IQtIZzBlock(short  *s ,int * d,short flag);
void GetYUV(short flag);
void StoreBuffer();
BYTE ReadByte();
void Initialize_Fast_IDCT();
void Fast_IDCT(int * block);
void idctrow(int * blk);
void idctcol(int * blk);
//////////////////////////////////////////////////
//global variable declaration
BITMAPFILEHEADER   bf;
BITMAPINFOHEADER   bi;
//HPALETTE           hPalette=NULL;
//HBITMAP            hBitmap=NULL;
char *            hImgData=NULL;
DWORD              NumColors;
DWORD              LineBytes;
DWORD              ImgWidth=0 , ImgHeight=0;
char*             lpPtr;
//////////////////////////////////////////////////
//variables used in jpeg function
short   SampRate_Y_H,SampRate_Y_V;
short   SampRate_U_H,SampRate_U_V;
short   SampRate_V_H,SampRate_V_V;
short   H_YtoU,V_YtoU,H_YtoV,V_YtoV;
short   Y_in_MCU,U_in_MCU,V_in_MCU;
unsigned char   *lpJpegBuf;
unsigned char   *lp;
short   qt_table[3][64];
short   comp_num;
BYTE   comp_index[3];
BYTE      YDcIndex,YAcIndex,UVDcIndex,UVAcIndex;
BYTE   HufTabIndex;
short      *YQtTable,*UQtTable,*VQtTable;
BYTE   And[9]={0,1,3,7,0xf,0x1f,0x3f,0x7f,0xff};
short      code_pos_table[4][16],code_len_table[4][16];
unsigned short code_value_table[4][256];
unsigned short huf_max_value[4][16],huf_min_value[4][16];
short   BitPos,CurByte;
short   rrun,vvalue;
short   MCUBuffer[10*64];
int    QtZzMCUBuffer[10*64];
short   BlockBuffer[64];
short   ycoef,ucoef,vcoef;
BOOL   IntervalFlag;
short   interval=0;
int    Y[4*64],U[4*64],V[4*64];
DWORD      sizei,sizej;
short    restart;
static  long iclip[1024];
static  long *iclp;

 

////////////////////////////////////////////////////////////////
BOOL LoadJpegFile (char *JpegFileName)
{
 FILE*      hfjpg;
 DWORD          ImgSize;
 DWORD              BufSize,JpegBufSize;
 FILE*              hfbmp;
 FILE*              IMGdata;
 void *      hJpegBuf;
 int       funcret;
 DWORD i;
 LPBITMAPINFOHEADER lpImgData;

 char * hImgData256;

 if((hfjpg=fopen(JpegFileName,"rb"))==NULL)
 {
  showerror(FUNC_FILE_ERROR);
  return FALSE;
 }
/*SEEK_CUR 
Current position of file pointer. 
SEEK_END 
End of file. 
SEEK_SET 
Beginning of file.*/


 //get jpg file length
 fseek(hfjpg,0L,SEEK_END);
 JpegBufSize=ftell(hfjpg);
 //rewind to the beginning of the file
 fseek(hfjpg,0L,SEEK_SET);

 if((hJpegBuf=malloc(JpegBufSize))==NULL)
 {
  fclose(hfjpg);
  showerror(FUNC_MEMORY_ERROR);
  
  return FALSE;
 }
 lpJpegBuf=(unsigned char  *)hJpegBuf;
 fread((unsigned char  *)hJpegBuf,sizeof( char ),JpegBufSize,hfjpg);
 fclose(hfjpg);
 
 InitTable();

 if((funcret=InitTag())!=FUNC_OK)
 {
 // GlobalUnlock(hJpegBuf);
  free(hJpegBuf);
  showerror(funcret);
  return FALSE;
 }
 //create new bitmapfileheader and bitmapinfoheader
 memset((char *)&bf,0,sizeof(BITMAPFILEHEADER)); 
 memset((char *)&bi,0,sizeof(BITMAPINFOHEADER));

 bi.biSize=(DWORD)sizeof(BITMAPINFOHEADER);
 bi.biWidth=(LONG)(ImgWidth);
 bi.biHeight=(LONG)(ImgHeight);
 bi.biPlanes=1;
 bi.biBitCount=24;
 bi.biClrUsed=0;
 bi.biClrImportant=0;
 bi.biCompression=BI_RGB;
 NumColors=0;
 printf("bi.biWidth is %ld/n",bi.biWidth);
printf("bi.biBitCount is %ld/n",bi.biBitCount);
 LineBytes=(DWORD)WIDTHBYTES(bi.biWidth*bi.biBitCount);
 printf("LineBytes is %ld/n",LineBytes);
 ImgSize=(DWORD)LineBytes*bi.biHeight;//???????
 printf("size is %ld/n",ImgSize);
 bf.bfType=0x4d42;
int a= sizeof(BITMAPFILEHEADER);
int b= sizeof(BITMAPINFOHEADER);
//注意字节对齐问题!!!!!!!!!!!!!!!!!!!!!!!!1
//如果没有#pragma pack(1),a是16~~~~~~~
int c=NumColors*sizeof(RGBQUAD);

 bf.bfSize=sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER)+NumColors*sizeof(RGBQUAD)+ImgSize;
 bf.bfOffBits=54;//(DWORD)(NumColors*sizeof(RGBQUAD)+sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER));
 BufSize=bf.bfSize-sizeof(BITMAPFILEHEADER);
// printf("size is %ld/n",BufSize);
 if((hImgData=(char*)malloc(BufSize))==NULL)
 {
  //GlobalUnlock(hJpegBuf);
  free(hJpegBuf);
  showerror(FUNC_MEMORY_ERROR);
  showerror(FUNC_MEMORY_ERROR);
  return FALSE;
 }
// lpImgData=(LPBITMAPINFOHEADER)GlobalLock(hImgData); 
 lpImgData=(LPBITMAPINFOHEADER)hImgData; 
 memcpy(lpImgData,(char *)&bi,sizeof(BITMAPINFOHEADER));
 lpPtr=(char *)lpImgData+sizeof(BITMAPINFOHEADER);

 if((SampRate_Y_H==0)||(SampRate_Y_V==0))
 {
 // GlobalUnlock(hJpegBuf);
  free(hJpegBuf);
  //GlobalUnlock(hImgData);
  free(hImgData);
  hImgData=NULL;
  showerror(FUNC_FORMAT_ERROR);
  return FALSE ;
 }

 funcret=Decode();
 if(funcret==FUNC_OK)
 {
  //hDc=GetDC(hWnd);
/*  hBitmap=CreateDIBitmap(hDc, (LPBITMAPINFOHEADER)lpImgData, (LONG)CBM_INIT,
      (LPSTR)lpImgData+sizeof(BITMAPINFOHEADER) +NumColors*sizeof(RGBQUAD),
         (LPBITMAPINFO)lpImgData, DIB_RGB_COLORS);
*/
 // hfbmp=fopen("c://jpeg2-bmp.bmp","Wb");

  if((hfbmp=fopen("F:\\data\\jpeg2-bmp.bmp","wb"))==NULL)
  {
   showerror(FUNC_FILE_ERROR);
   return FALSE;
  }

  // hfbmp=fopen("c://jpeg2-bmp.bmp","Wb");

 

  fwrite((LPSTR)&bf,sizeof(BITMAPFILEHEADER),1,hfbmp); 
  fwrite((LPSTR)lpImgData,sizeof(char),BufSize,hfbmp);

  if((IMGdata=fopen("111.txt","wb"))==NULL)
  {
   showerror(FUNC_FILE_ERROR);
   return FALSE;
  }
  DWORD xx = ImgWidth*ImgHeight;
  if(( hImgData256=(char *)malloc(xx))==NULL)
  {
   //GlobalUnlock(hJpegBuf);
   free(hImgData256);
   showerror(FUNC_MEMORY_ERROR);
   showerror(FUNC_MEMORY_ERROR);
   showerror(FUNC_MEMORY_ERROR);
   return FALSE;
  }

  char * temp=hImgData256;
  for(i=0; i < xx; i++)
  {
   i;
   char t3=*lpPtr;
   t3 &= 0xE0;
   char t1=*(lpPtr+1);
   t1 = t1>>3;
   t1 &= 0x1c;
   char t2=*(lpPtr+2);
   t2=t2>>6;
   t2 &= 0x03;
   char t4 = t3 + t1 + t2 ;
   *temp++=t4;
   lpPtr=lpPtr+ 3;
   //不能使用temp+=3;
  }
  int count=fwrite(hImgData256,sizeof(char),xx,IMGdata);
  //free((void*)hImgData256);
  fclose(IMGdata);

  fclose(hfbmp);
  
//  ReleaseDC(hWnd,hDc);
//  GlobalUnlock(hJpegBuf);
  free(hJpegBuf);
  //GlobalUnlock(hImgData);
  return TRUE;
 }
 else
 {
  //GlobalUnlock(hJpegBuf);
  free(hJpegBuf);
 // GlobalUnlock(hImgData);
  free(hImgData);
  hImgData=NULL;
  showerror(funcret);
  return FALSE;
 }
}
/////////////////////////////////////////////////
void showerror(int funcret)
{
 switch(funcret)
 {
 case FUNC_MEMORY_ERROR:
     printf("Error alloc memory/n!");
  break;
 case FUNC_FILE_ERROR:
     printf("File not found!/n");
  break;
 case FUNC_FORMAT_ERROR:
  printf("File format error!/n");
  break;
 }
}
////////////////////////////////////////////////////////////////////////////////
int InitTag()
{
 BOOL finish=FALSE;
 BYTE id;
 short  llength;
 short  i,j,k;
 short  huftab1,huftab2;
 short  huftabindex;
 BYTE hf_table_index;
 BYTE qt_table_index;
 BYTE comnum;

 unsigned char  *lptemp;
 short  ccount;

 lp=lpJpegBuf+2;

 while (!finish){
  id=*(lp+1);
  lp+=2;
  switch (id){
  case M_APP0:
   llength=MAKEWORD(*(lp+1),*lp);
   lp+=llength;
   break;
  case M_DQT:
   llength=MAKEWORD(*(lp+1),*lp);
   qt_table_index=(*(lp+2))&0x0f;
   lptemp=lp+3;
   if(llength<80){
    for(i=0;i<64;i++)
     qt_table[qt_table_index][i]=(short)*(lptemp++);
   }
   else{
    for(i=0;i<64;i++)
     qt_table[qt_table_index][i]=(short)*(lptemp++);
                qt_table_index=(*(lptemp++))&0x0f;
      for(i=0;i<64;i++)
     qt_table[qt_table_index][i]=(short)*(lptemp++);
     }
     lp+=llength;  
   break;
  case M_SOF0:
    llength=MAKEWORD(*(lp+1),*lp);
    ImgHeight=MAKEWORD(*(lp+4),*(lp+3));
    ImgWidth=MAKEWORD(*(lp+6),*(lp+5));
            comp_num=*(lp+7);
      if((comp_num!=1)&&(comp_num!=3))
      return FUNC_FORMAT_ERROR;
   if(comp_num==3){
    comp_index[0]=*(lp+8);
      SampRate_Y_H=(*(lp+9))>>4;
      SampRate_Y_V=(*(lp+9))&0x0f;
      YQtTable=(short *)qt_table[*(lp+10)];

    comp_index[1]=*(lp+11);
    SampRate_U_H=(*(lp+12))>>4;
      SampRate_U_V=(*(lp+12))&0x0f;
      UQtTable=(short *)qt_table[*(lp+13)];

      comp_index[2]=*(lp+14);
      SampRate_V_H=(*(lp+15))>>4;
      SampRate_V_V=(*(lp+15))&0x0f;
    VQtTable=(short *)qt_table[*(lp+16)];
     }
   else{
      comp_index[0]=*(lp+8);
    SampRate_Y_H=(*(lp+9))>>4;
      SampRate_Y_V=(*(lp+9))&0x0f;
      YQtTable=(short *)qt_table[*(lp+10)];

    comp_index[1]=*(lp+8);
      SampRate_U_H=1;
      SampRate_U_V=1;
      UQtTable=(short *)qt_table[*(lp+10)];

    comp_index[2]=*(lp+8);
    SampRate_V_H=1;
      SampRate_V_V=1;
      VQtTable=(short *)qt_table[*(lp+10)];
   }
     lp+=llength;          
   break;
  case M_DHT:             
   llength=MAKEWORD(*(lp+1),*lp);
   if (llength<0xd0){
    huftab1=(short)(*(lp+2))>>4;     //huftab1=0,1
     huftab2=(short)(*(lp+2))&0x0f;   //huftab2=0,1
    huftabindex=huftab1*2+huftab2;
     lptemp=lp+3;
    for (i=0; i<16; i++)
     code_len_table[huftabindex][i]=(short)(*(lptemp++));
    j=0;
    for (i=0; i<16; i++)
     if(code_len_table[huftabindex][i]!=0){
      k=0;
      while(k<code_len_table[huftabindex][i]){
       code_value_table[huftabindex][k+j]=(short)(*(lptemp++));
       k++;
      }
      j+=k; 
     }
    i=0;
    while (code_len_table[huftabindex][i]==0)
      i++;
    for (j=0;j<i;j++){
     huf_min_value[huftabindex][j]=0;
     huf_max_value[huftabindex][j]=0;
    }
    huf_min_value[huftabindex][i]=0;
    huf_max_value[huftabindex][i]=code_len_table[huftabindex][i]-1;
    for (j=i+1;j<16;j++){
     huf_min_value[huftabindex][j]=(huf_max_value[huftabindex][j-1]+1)<<1;
     huf_max_value[huftabindex][j]=huf_min_value[huftabindex][j]+code_len_table[huftabindex][j]-1;
    }
    code_pos_table[huftabindex][0]=0;
    for (j=1;j<16;j++)
       code_pos_table[huftabindex][j]=code_len_table[huftabindex][j-1]+code_pos_table[huftabindex][j-1];
      lp+=llength;
   }  //if
   else{
     hf_table_index=*(lp+2);
    lp+=2;
    while (hf_table_index!=0xff){
     huftab1=(short)hf_table_index>>4;     //huftab1=0,1
      huftab2=(short)hf_table_index&0x0f;   //huftab2=0,1
     huftabindex=huftab1*2+huftab2;
     lptemp=lp+1;
     ccount=0;
     for (i=0; i<16; i++){
      code_len_table[huftabindex][i]=(short)(*(lptemp++));
      ccount+=code_len_table[huftabindex][i];
     }
     ccount+=17; 
     j=0;
     for (i=0; i<16; i++)
      if(code_len_table[huftabindex][i]!=0){
       k=0;
       while(k<code_len_table[huftabindex][i])
       {
        code_value_table[huftabindex][k+j]=(short)(*(lptemp++));
        k++;
       }
       j+=k;
      }
     i=0;
     while (code_len_table[huftabindex][i]==0)
      i++;
     for (j=0;j<i;j++){
      huf_min_value[huftabindex][j]=0;
      huf_max_value[huftabindex][j]=0;
     }
     huf_min_value[huftabindex][i]=0;
     huf_max_value[huftabindex][i]=code_len_table[huftabindex][i]-1;
     for (j=i+1;j<16;j++){
      huf_min_value[huftabindex][j]=(huf_max_value[huftabindex][j-1]+1)<<1;
      huf_max_value[huftabindex][j]=huf_min_value[huftabindex][j]+code_len_table[huftabindex][j]-1;
     }
     code_pos_table[huftabindex][0]=0;
     for (j=1;j<16;j++)
      code_pos_table[huftabindex][j]=code_len_table[huftabindex][j-1]+code_pos_table[huftabindex][j-1];
     lp+=ccount;
     hf_table_index=*lp;
    }  //while
   }  //else
   break;
  case M_DRI:
   llength=MAKEWORD(*(lp+1),*lp);
   restart=MAKEWORD(*(lp+3),*(lp+2));
   lp+=llength;
   break;
  case M_SOS:
   llength=MAKEWORD(*(lp+1),*lp);
   comnum=*(lp+2);
   if(comnum!=comp_num)
    return FUNC_FORMAT_ERROR;
   lptemp=lp+3;
   for (i=0;i<comp_num;i++){
    if(*lptemp==comp_index[0]){
     YDcIndex=(*(lptemp+1))>>4;   //Y
     YAcIndex=((*(lptemp+1))&0x0f)+2;
    }
    else{
     UVDcIndex=(*(lptemp+1))>>4;   //U,V
     UVAcIndex=((*(lptemp+1))&0x0f)+2;
    }
    lptemp+=2;
   }
   lp+=llength;
   finish=TRUE;
   break;
  case M_EOI:    
   return FUNC_FORMAT_ERROR;
   break;
  default:
    if ((id&0xf0)!=0xd0){
    llength=MAKEWORD(*(lp+1),*lp);
     lp+=llength;
   }
   else lp+=2;
   break;
    }  //switch
 } //while
 return FUNC_OK;
}
/////////////////////////////////////////////////////////////////
void InitTable()
{
 short i,j;
 sizei=sizej=0;
 ImgWidth=ImgHeight=0;
 rrun=vvalue=0;
 BitPos=0;
 CurByte=0;
 IntervalFlag=FALSE;
 restart=0;
 for(i=0;i<3;i++)
  for(j=0;j<64;j++)
   qt_table[i][j]=0;
 comp_num=0;
 HufTabIndex=0;
 for(i=0;i<3;i++)
  comp_index[i]=0;
 for(i=0;i<4;i++)
  for(j=0;j<16;j++){
   code_len_table[i][j]=0;
   code_pos_table[i][j]=0;
   huf_max_value[i][j]=0;
   huf_min_value[i][j]=0;
  }
 for(i=0;i<4;i++)
  for(j=0;j<256;j++)
   code_value_table[i][j]=0;
 
 for(i=0;i<10*64;i++){
  MCUBuffer[i]=0;
  QtZzMCUBuffer[i]=0;
 }
 for(i=0;i<64;i++){
  Y[i]=0;
  U[i]=0;
  V[i]=0;
  BlockBuffer[i]=0;
 }
 ycoef=ucoef=vcoef=0;
}
/////////////////////////////////////////////////////////////////////////
int Decode()
{
 int funcret;

 Y_in_MCU=SampRate_Y_H*SampRate_Y_V;
 U_in_MCU=SampRate_U_H*SampRate_U_V;
 V_in_MCU=SampRate_V_H*SampRate_V_V;
 H_YtoU=SampRate_Y_H/SampRate_U_H;
 V_YtoU=SampRate_Y_V/SampRate_U_V;
 H_YtoV=SampRate_Y_H/SampRate_V_H;
 V_YtoV=SampRate_Y_V/SampRate_V_V;
 Initialize_Fast_IDCT();
 while((funcret=DecodeMCUBlock())==FUNC_OK){
  interval++;
  if((restart)&&(interval % restart==0))
    IntervalFlag=TRUE;
  else
   IntervalFlag=FALSE;
  IQtIZzMCUComponent(0);
  IQtIZzMCUComponent(1);
  IQtIZzMCUComponent(2);
  GetYUV(0);
  GetYUV(1);
  GetYUV(2);
  StoreBuffer();
  sizej+=SampRate_Y_H*8;
  if(sizej>=ImgWidth){
   sizej=0;
   sizei+=SampRate_Y_V*8;
  }
  if ((sizej==0)&&(sizei>=ImgHeight))
   break;
  }
 return funcret;
}
/////////////////////////////////////////////////////////////////////////////////////////
void  GetYUV(short flag)
{
 short H,VV;
 short i,j,k,h;
 int  *buf;
 int  *pQtZzMCU;

 switch(flag){
 case 0:
  H=SampRate_Y_H;
  VV=SampRate_Y_V;
  buf=Y;
  pQtZzMCU=QtZzMCUBuffer;
  break;
 case 1:
  H=SampRate_U_H;
  VV=SampRate_U_V;
  buf=U;
  pQtZzMCU=QtZzMCUBuffer+Y_in_MCU*64;
  break;
 case 2:
  H=SampRate_V_H;
  VV=SampRate_V_V;
  buf=V;
  pQtZzMCU=QtZzMCUBuffer+(Y_in_MCU+U_in_MCU)*64;
  break;
 }
 for (i=0;i<VV;i++)
  for(j=0;j<H;j++)
   for(k=0;k<8;k++)
    for(h=0;h<8;h++)
     buf[(i*8+k)*SampRate_Y_H*8+j*8+h]=*pQtZzMCU++;
}
///////////////////////////////////////////////////////////////////////////////
void StoreBuffer()
{
 short i,j;
 unsigned char  *lpbmp;
 unsigned char R,G,B;
 int y,u,v,rr,gg,bb;

 for(i=0;i<SampRate_Y_V*8;i++){
  if((sizei+i)<ImgHeight){
   lpbmp=((unsigned char *)lpPtr+(DWORD)(ImgHeight-sizei-i-1)*LineBytes+sizej*3);
   for(j=0;j<SampRate_Y_H*8;j++){
    if((sizej+j)<ImgWidth){
     y=Y[i*8*SampRate_Y_H+j];
     u=U[(i/V_YtoU)*8*SampRate_Y_H+j/H_YtoU];
     v=V[(i/V_YtoV)*8*SampRate_Y_H+j/H_YtoV];
     rr=((y<<8)+18*u+367*v)>>8;
     gg=((y<<8)-159*u-220*v)>>8;
     bb=((y<<8)+411*u-29*v)>>8;
     R=(unsigned char)rr;
     G=(unsigned char)gg;
     B=(unsigned char)bb;
     if (rr&0xffffff00) if (rr>255) R=255; else if (rr<0) R=0;
     if (gg&0xffffff00) if (gg>255) G=255; else if (gg<0) G=0;
     if (bb&0xffffff00) if (bb>255) B=255; else if (bb<0) B=0;
     *lpbmp++=B;
     *lpbmp++=G;
     *lpbmp++=R;
     
     
    }
    else  break;
   }
  }
  else break;
 }
}
///////////////////////////////////////////////////////////////////////////////
int DecodeMCUBlock()
{
 short *lpMCUBuffer;
 short i,j;
 int funcret;

 if (IntervalFlag){
  lp+=2;
  ycoef=ucoef=vcoef=0;
  BitPos=0;
  CurByte=0;
 }
 switch(comp_num){
 case 3:
  lpMCUBuffer=MCUBuffer;
  for (i=0;i<SampRate_Y_H*SampRate_Y_V;i++)  //Y
  {
   funcret=HufBlock(YDcIndex,YAcIndex);
   if (funcret!=FUNC_OK)
    return funcret;
   BlockBuffer[0]=BlockBuffer[0]+ycoef;
   ycoef=BlockBuffer[0];
   for (j=0;j<64;j++)
    *lpMCUBuffer++=BlockBuffer[j];
  }
  for (i=0;i<SampRate_U_H*SampRate_U_V;i++)  //U
  {
   funcret=HufBlock(UVDcIndex,UVAcIndex);
   if (funcret!=FUNC_OK)
    return funcret;
   BlockBuffer[0]=BlockBuffer[0]+ucoef;
   ucoef=BlockBuffer[0];
   for (j=0;j<64;j++)
    *lpMCUBuffer++=BlockBuffer[j];
  }
  for (i=0;i<SampRate_V_H*SampRate_V_V;i++)  //V
  {
   funcret=HufBlock(UVDcIndex,UVAcIndex);
   if (funcret!=FUNC_OK)
    return funcret;
   BlockBuffer[0]=BlockBuffer[0]+vcoef;
   vcoef=BlockBuffer[0];
   for (j=0;j<64;j++)
    *lpMCUBuffer++=BlockBuffer[j];
  }
  break;
 case 1:
  lpMCUBuffer=MCUBuffer;
  funcret=HufBlock(YDcIndex,YAcIndex);
  if (funcret!=FUNC_OK)
   return funcret;
  BlockBuffer[0]=BlockBuffer[0]+ycoef;
  ycoef=BlockBuffer[0];
  for (j=0;j<64;j++)
   *lpMCUBuffer++=BlockBuffer[j];
  for (i=0;i<128;i++)
   *lpMCUBuffer++=0;
  break;
 default:
  return FUNC_FORMAT_ERROR;
 }
 return FUNC_OK;
}
//////////////////////////////////////////////////////////////////
int HufBlock(BYTE dchufindex,BYTE achufindex)
{
 short count=0;
 short i;
 int funcret;

 //dc
 HufTabIndex=dchufindex;
 funcret=DecodeElement();
 if(funcret!=FUNC_OK)
  return funcret;

 BlockBuffer[count++]=vvalue;
 //ac
 HufTabIndex=achufindex;
 while (count<64){
  funcret=DecodeElement();
  if(funcret!=FUNC_OK)
   return funcret;
  if ((rrun==0)&&(vvalue==0)){
   for (i=count;i<64;i++)
    BlockBuffer[i]=0;
   count=64;
  }
  else{
   for (i=0;i<rrun;i++)
    BlockBuffer[count++]=0;
   BlockBuffer[count++]=vvalue;
  }
 }
 return FUNC_OK;
}
//////////////////////////////////////////////////////////////////////////////
int DecodeElement()
{
 int thiscode,tempcode;
 unsigned short temp,valueex;
 short codelen;
 BYTE hufexbyte,runsize,tempsize,sign;
 BYTE newbyte,lastbyte;

 if(BitPos>=1){
  BitPos--;
  thiscode=(BYTE)CurByte>>BitPos;
  CurByte=CurByte&And[BitPos];
 }
 else{
  lastbyte=ReadByte();
  BitPos--;
  newbyte=CurByte&And[BitPos];
  thiscode=lastbyte>>7;
  CurByte=newbyte;
 }
 codelen=1;
 while ((thiscode<huf_min_value[HufTabIndex][codelen-1])||
    (code_len_table[HufTabIndex][codelen-1]==0)||
    (thiscode>huf_max_value[HufTabIndex][codelen-1]))
 {
  if(BitPos>=1){
   BitPos--;
   tempcode=(BYTE)CurByte>>BitPos;
   CurByte=CurByte&And[BitPos];
  }
  else{
   lastbyte=ReadByte();
   BitPos--;
   newbyte=CurByte&And[BitPos];
   tempcode=(BYTE)lastbyte>>7;
   CurByte=newbyte;
  }
  thiscode=(thiscode<<1)+tempcode;
  codelen++;
  if(codelen>16)
   return FUNC_FORMAT_ERROR;
 }  //while
 temp=thiscode-huf_min_value[HufTabIndex][codelen-1]+code_pos_table[HufTabIndex][codelen-1];
 hufexbyte=(BYTE)code_value_table[HufTabIndex][temp];
 rrun=(short)(hufexbyte>>4);
 runsize=hufexbyte&0x0f;
 if(runsize==0){
  vvalue=0;
  return FUNC_OK;
 }
 tempsize=runsize;
 if(BitPos>=runsize){
  BitPos-=runsize;
  valueex=(BYTE)CurByte>>BitPos;
  CurByte=CurByte&And[BitPos];
 }
 else{
  valueex=CurByte;
  tempsize-=BitPos;
  while(tempsize>8){
   lastbyte=ReadByte();
   valueex=(valueex<<8)+(BYTE)lastbyte;
   tempsize-=8;
  }  //while
  lastbyte=ReadByte();
  BitPos-=tempsize;
  valueex=(valueex<<tempsize)+(lastbyte>>BitPos);
  CurByte=lastbyte&And[BitPos];
 }  //else
 sign=valueex>>(runsize-1);
 if(sign)
  vvalue=valueex;
 else{
  valueex=valueex^0xffff;
  temp=0xffff<<runsize;
  vvalue=-(short)(valueex^temp);
 }
 return FUNC_OK;
}
/////////////////////////////////////////////////////////////////////////////////////
void IQtIZzMCUComponent(short flag)
{
 short H,VV;
 short i,j;
 int *pQtZzMCUBuffer;
 short  *pMCUBuffer;

 switch(flag){
 case 0:
  H=SampRate_Y_H;
  VV=SampRate_Y_V;
  pMCUBuffer=MCUBuffer;
  pQtZzMCUBuffer=QtZzMCUBuffer;
  break;
 case 1:
  H=SampRate_U_H;
  VV=SampRate_U_V;
  pMCUBuffer=MCUBuffer+Y_in_MCU*64;
  pQtZzMCUBuffer=QtZzMCUBuffer+Y_in_MCU*64;
  break;
 case 2:
  H=SampRate_V_H;
  VV=SampRate_V_V;
  pMCUBuffer=MCUBuffer+(Y_in_MCU+U_in_MCU)*64;
  pQtZzMCUBuffer=QtZzMCUBuffer+(Y_in_MCU+U_in_MCU)*64;
  break;
 }
 for(i=0;i<VV;i++)
  for (j=0;j<H;j++)
   IQtIZzBlock(pMCUBuffer+(i*H+j)*64,pQtZzMCUBuffer+(i*H+j)*64,flag);
}
//////////////////////////////////////////////////////////////////////////////////////////
void IQtIZzBlock(short  *s ,int * d,short flag)
{
 short i,j;
 short tag;
 short *pQt;
 int buffer2[8][8];
 int *buffer1;
 short offset;

 switch(flag){
 case 0:
  pQt=YQtTable;
  offset=128;
  break;
 case 1:
  pQt=UQtTable;
  offset=0;
  break;
 case 2:
  pQt=VQtTable;
  offset=0;
  break;
 }

 for(i=0;i<8;i++)
  for(j=0;j<8;j++){
   tag=Zig_Zag[i][j];
   buffer2[i][j]=(int)s[tag]*(int)pQt[tag];
  }
 buffer1=(int *)buffer2;
 Fast_IDCT(buffer1);
 for(i=0;i<8;i++)
  for(j=0;j<8;j++)
   d[i*8+j]=buffer2[i][j]+offset;
}
///////////////////////////////////////////////////////////////////////////////
void Fast_IDCT(int * block)
{
 short i;

 for (i=0; i<8; i++)
  idctrow(block+8*i);

 for (i=0; i<8; i++)
  idctcol(block+i);
}
///////////////////////////////////////////////////////////////////////////////
BYTE  ReadByte()
{
 BYTE  i;

 i=*(lp++);
 if(i==0xff)
  lp++;
 BitPos=8;
 CurByte=i;
 return i;
}
///////////////////////////////////////////////////////////////////////
void Initialize_Fast_IDCT()
{
 short i;

 iclp = iclip+512;
 for (i= -512; i<512; i++)
  iclp[i] = (i<-256) ? -256 : ((i>255) ? 255 : i);
}
////////////////////////////////////////////////////////////////////////
void idctrow(int * blk)
{
 int x0, x1, x2, x3, x4, x5, x6, x7, x8;
 //intcut
 if (!((x1 = blk[4]<<11) | (x2 = blk[6]) | (x3 = blk[2]) |
  (x4 = blk[1]) | (x5 = blk[7]) | (x6 = blk[5]) | (x7 = blk[3])))
 {
  blk[0]=blk[1]=blk[2]=blk[3]=blk[4]=blk[5]=blk[6]=blk[7]=blk[0]<<3;
  return;
 }
 x0 = (blk[0]<<11) + 128; // for proper rounding in the fourth stage 
 //first stage
 x8 = W7*(x4+x5);
 x4 = x8 + (W1-W7)*x4;
 x5 = x8 - (W1+W7)*x5;
 x8 = W3*(x6+x7);
 x6 = x8 - (W3-W5)*x6;
 x7 = x8 - (W3+W5)*x7;
 //second stage
 x8 = x0 + x1;
 x0 -= x1;
 x1 = W6*(x3+x2);
 x2 = x1 - (W2+W6)*x2;
 x3 = x1 + (W2-W6)*x3;
 x1 = x4 + x6;
 x4 -= x6;
 x6 = x5 + x7;
 x5 -= x7;
 //third stage
 x7 = x8 + x3;
 x8 -= x3;
 x3 = x0 + x2;
 x0 -= x2;
 x2 = (181*(x4+x5)+128)>>8;
 x4 = (181*(x4-x5)+128)>>8;
 //fourth stage
 blk[0] = (x7+x1)>>8;
 blk[1] = (x3+x2)>>8;
 blk[2] = (x0+x4)>>8;
 blk[3] = (x8+x6)>>8;
 blk[4] = (x8-x6)>>8;
 blk[5] = (x0-x4)>>8;
 blk[6] = (x3-x2)>>8;
 blk[7] = (x7-x1)>>8;
}
//////////////////////////////////////////////////////////////////////////////
void idctcol(int * blk)
{
 int x0, x1, x2, x3, x4, x5, x6, x7, x8;
 //intcut
 if (!((x1 = (blk[8*4]<<8)) | (x2 = blk[8*6]) | (x3 = blk[8*2]) |
  (x4 = blk[8*1]) | (x5 = blk[8*7]) | (x6 = blk[8*5]) | (x7 = blk[8*3])))
 {
  blk[8*0]=blk[8*1]=blk[8*2]=blk[8*3]=blk[8*4]=blk[8*5]
   =blk[8*6]=blk[8*7]=iclp[(blk[8*0]+32)>>6];
  return;
 }
 x0 = (blk[8*0]<<8) + 8192;
 //first stage
 x8 = W7*(x4+x5) + 4;
 x4 = (x8+(W1-W7)*x4)>>3;
 x5 = (x8-(W1+W7)*x5)>>3;
 x8 = W3*(x6+x7) + 4;
 x6 = (x8-(W3-W5)*x6)>>3;
 x7 = (x8-(W3+W5)*x7)>>3;
 //second stage
 x8 = x0 + x1;
 x0 -= x1;
 x1 = W6*(x3+x2) + 4;
 x2 = (x1-(W2+W6)*x2)>>3;
 x3 = (x1+(W2-W6)*x3)>>3;
 x1 = x4 + x6;
 x4 -= x6;
 x6 = x5 + x7;
 x5 -= x7;
 //third stage
 x7 = x8 + x3;
 x8 -= x3;
 x3 = x0 + x2;
 x0 -= x2;
 x2 = (181*(x4+x5)+128)>>8;
 x4 = (181*(x4-x5)+128)>>8;
 //fourth stage
 blk[8*0] = iclp[(x7+x1)>>14];
 blk[8*1] = iclp[(x3+x2)>>14];
 blk[8*2] = iclp[(x0+x4)>>14];
 blk[8*3] = iclp[(x8+x6)>>14];
 blk[8*4] = iclp[(x8-x6)>>14];
 blk[8*5] = iclp[(x0-x4)>>14];
 blk[8*6] = iclp[(x3-x2)>>14];
 blk[8*7] = iclp[(x7-x1)>>14];
}

int _tmain(int argc, _TCHAR* argv[])
{
	LoadJpegFile("F:\\data\\IMG_20150411_183508.jpg");

	return 0;
}


你可能感兴趣的:(简单的JPEG解码程序)