/* Pcsx2 - Pc Ps2 Emulator * Copyright (C) 2002-2008 Pcsx2 Team * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ /* This code was based on the FPSE v0.08 Mdec decoder*/ #if 0 #include #include #include "IopCommon.h" #include "Mdec.h" int iq_y[DCTSIZE2],iq_uv[DCTSIZE2]; static void idct1(int *block) { int i, val; val = RANGE(DESCALE(block[0], PASS1_BITS+3)); for(i=0;i>16)*(bcr&0xffff); if (cmd==0x40000001) { u8 *p = (u8*)PSXM(adr); iqtab_init(iq_y,p); iqtab_init(iq_uv,p+64); } else if ((cmd&0xf5ff0000)==0x30000000) { mdec.rl = (u16*)PSXM(adr); } HW_DMA0_CHCR &= ~0x01000000; psxDmaInterrupt(0); } void psxDma1(u32 adr, u32 bcr, u32 chcr) { int blk[DCTSIZE2*6]; unsigned short *image; int size; CDR_LOG("DMA1 %lx %lx %lx (cmd = %lx)", adr, bcr, chcr, mdec.command); if (chcr!=0x01000200) return; size = (bcr>>16)*(bcr&0xffff); image = (u16*)PSXM(adr); if (mdec.command&0x08000000) { for (;size>0;size-=(16*16)/2,image+=(16*16)) { mdec.rl = rl2blk(blk,mdec.rl); yuv2rgb15(blk,image); } } else { for (;size>0;size-=(24*16)/2,image+=(24*16)) { mdec.rl = rl2blk(blk,mdec.rl); yuv2rgb24(blk,(u8 *)image); } } HW_DMA1_CHCR &= ~0x01000000; psxDmaInterrupt(1); } static int zscan[DCTSIZE2] = { 0 ,1 ,8 ,16,9 ,2 ,3 ,10, 17,24,32,25,18,11,4 ,5 , 12,19,26,33,40,48,41,34, 27,20,13,6 ,7 ,14,21,28, 35,42,49,56,57,50,43,36, 29,22,15,23,30,37,44,51, 58,59,52,45,38,31,39,46, 53,60,61,54,47,55,62,63 }; static int aanscales[DCTSIZE2] = { 16384, 22725, 21407, 19266, 16384, 12873, 8867, 4520, 22725, 31521, 29692, 26722, 22725, 17855, 12299, 6270, 21407, 29692, 27969, 25172, 21407, 16819, 11585, 5906, 19266, 26722, 25172, 22654, 19266, 15137, 10426, 5315, 16384, 22725, 21407, 19266, 16384, 12873, 8867, 4520, 12873, 17855, 16819, 15137, 12873, 10114, 6967, 3552, 8867, 12299, 11585, 10426, 8867, 6967, 4799, 2446, 4520, 6270, 5906, 5315, 4520, 3552, 2446, 1247 }; void iqtab_init(int *iqtab,unsigned char *iq_y) { int i; for(i=0;i>(CONST_BITS14-IFAST_SCALE_BITS); } } unsigned short* rl2blk(int *blk,unsigned short *mdec_rl) { int i,k,q_scale,rl; int *iqtab; memset (blk, 0, 6*DCTSIZE2*4); iqtab = iq_uv; for(i=0;i<6;i++) { // decode blocks (Cr,Cb,Y1,Y2,Y3,Y4) if (i>1) iqtab = iq_y; // zigzag transformation rl = *mdec_rl++; q_scale = RUNOF(rl); blk[0] = iqtab[0]*VALOF(rl); for(k = 0;;) { rl = *mdec_rl++; if (rl==NOP) break; k += RUNOF(rl)+1; // skip level zero-coefficients if (k > 63) break; blk[zscan[k]] = (VALOF(rl) * iqtab[k] * q_scale) / 8; // / 16; } idct(blk,k+1); blk+=DCTSIZE2; } return mdec_rl; } unsigned char roundtbl[256*3]; void round_init(void) { int i; for(i=0;i<256;i++) { roundtbl[i]=0; roundtbl[i+256]=i; roundtbl[i+512]=255; } } void yuv2rgb15(int *blk,unsigned short *image) { int x,y; int *Yblk = blk+DCTSIZE2*2; int Cb,Cr,R,G,B; int *Cbblk = blk; int *Crblk = blk+DCTSIZE2; if (!(Config.Mdec&0x1)) for (y=0;y<16;y+=2,Crblk+=4,Cbblk+=4,Yblk+=8,image+=24) { if (y==8) Yblk+=DCTSIZE2; for (x=0;x<4;x++,image+=2,Crblk++,Cbblk++,Yblk+=2) { Cr = *Crblk; Cb = *Cbblk; R = MULR(Cr); G = MULG(Cb) + MULG2(Cr); B = MULB(Cb); RGB15(0, Yblk[0]); RGB15(1, Yblk[1]); RGB15(16, Yblk[8]); RGB15(17, Yblk[9]); Cr = *(Crblk+4); Cb = *(Cbblk+4); R = MULR(Cr); G = MULG(Cb) + MULG2(Cr); B = MULB(Cb); RGB15(8, Yblk[DCTSIZE2+0]); RGB15(9, Yblk[DCTSIZE2+1]); RGB15(24, Yblk[DCTSIZE2+8]); RGB15(25, Yblk[DCTSIZE2+9]); } } else for (y=0;y<16;y+=2,Yblk+=8,image+=24) { if (y==8) Yblk+=DCTSIZE2; for (x=0;x<4;x++,image+=2,Yblk+=2) { RGB15BW(0, Yblk[0]); RGB15BW(1, Yblk[1]); RGB15BW(16, Yblk[8]); RGB15BW(17, Yblk[9]); RGB15BW(8, Yblk[DCTSIZE2+0]); RGB15BW(9, Yblk[DCTSIZE2+1]); RGB15BW(24, Yblk[DCTSIZE2+8]); RGB15BW(25, Yblk[DCTSIZE2+9]); } } } void yuv2rgb24(int *blk,unsigned char *image) { int x,y; int *Yblk = blk+DCTSIZE2*2; int Cb,Cr,R,G,B; int *Cbblk = blk; int *Crblk = blk+DCTSIZE2; if (!(Config.Mdec&0x1)) for (y=0;y<16;y+=2,Crblk+=4,Cbblk+=4,Yblk+=8,image+=24*3) { if (y==8) Yblk+=DCTSIZE2; for (x=0;x<4;x++,image+=6,Crblk++,Cbblk++,Yblk+=2) { Cr = *Crblk; Cb = *Cbblk; R = MULR(Cr); G = MULG(Cb) + MULG2(Cr); B = MULB(Cb); RGB24(0, Yblk[0]); RGB24(1*3, Yblk[1]); RGB24(16*3, Yblk[8]); RGB24(17*3, Yblk[9]); Cr = *(Crblk+4); Cb = *(Cbblk+4); R = MULR(Cr); G = MULG(Cb) + MULG2(Cr); B = MULB(Cb); RGB24(8*3, Yblk[DCTSIZE2+0]); RGB24(9*3, Yblk[DCTSIZE2+1]); RGB24(24*3, Yblk[DCTSIZE2+8]); RGB24(25*3, Yblk[DCTSIZE2+9]); } } else for (y=0;y<16;y+=2,Yblk+=8,image+=24*3) { if (y==8) Yblk+=DCTSIZE2; for (x=0;x<4;x++,image+=6,Yblk+=2) { RGB24BW(0, Yblk[0]); RGB24BW(1*3, Yblk[1]); RGB24BW(16*3, Yblk[8]); RGB24BW(17*3, Yblk[9]); RGB24BW(8*3, Yblk[DCTSIZE2+0]); RGB24BW(9*3, Yblk[DCTSIZE2+1]); RGB24BW(24*3, Yblk[DCTSIZE2+8]); RGB24BW(25*3, Yblk[DCTSIZE2+9]); } } } int SaveState::mdecFreeze() { Freeze(mdec); Freeze(iq_y); Freeze(iq_uv); return 0; } #endif