GBA探索日记(-)

    技术2022-05-11  140

    GBA探索日记(-)

    mode0-2的BG与VRAM

     

    GBAVRAM总共96KB,在mode3-5中,VRAM跟计算机的显示内存一样.屏幕上一个点对应一个显示内存地址.

    可以看看下面两个函数,都是在mode4下的写点函数.

    void PlotPixel(int x,int y, unsigned short int c)

    {

                    m_VideoBuffer[(y) * 120 + (x)] = (c);

    }

     

    void SetPixel(int x,int y,u8 color)

    {

                    register u16 *tc;

                tc=m_VideoBuffer+y*120+x/2;

                    if(x&1)

                                   *tc=((*tc&0x00FF)+(color<<8));

                else

                              *tc=(*tc&0xFF00)+color;

     

    }

    由于GBA支持的最小数据传输单位为16位,而16位在mode4(256色)下为两个点的信息.

    所以上面的PlotPixel是对两个点的写法.而SetPixel是通过一些简单的位移运算来实现对单个点的写入.

    特别指出的是一个VRAM里的u16的值低8位对应的是前面的点,高8位对应的是后面的点.

     

    说了这么多mode4的,下面看看GBA最精彩的部分mode0-2

     

    设置REG_DISPCNT这个寄存器

    不说多了,它就是指定你要显示的模式和支持的BG.

    比如看这个的语句:

    *(vu16 *)REG_DISPCNT = DISP_MODE_1 |  DISP_BG0_ON ;

    这段语句就是说让GBA支持MODE1的显示模式,并支持BG0

    当然你也可以

    *(vu16 *)REG_DISPCNT = DISP_MODE_1 |  DISP_BG0_ON | DISP_BG1_ON |DISP_BG2_ON;

    但是你不能把BG3也加进来.因为mode1只能支持BG0,BG1,BG2.

     

     

    mode0-2都是运用tile(图块)先来组成整个屏幕的.

    mode0-2每个都有不同的BG

    一个屏幕可以由多个BG组合显示出来.比如你可以安排地面为一个BG,天空为一个BG等等.

    每个BG对应一个寄存器

    Text BG:就是不能旋转的,似乎只能使用16色.当然,如果使用16色的调色板,当然可以使用最多到16个调色板.

    Rotation BG:就是可以旋转的,放大的.只能使用256色.如果使用了256色的调色板,就不能使用16色的调色板了.因为GBA的BG调色板就只有256x2=512那么大.

     

     

    再看看不同模式下的BG分配

    mode 0: 4个Text BG

    mode 1: Text BG:        BG0 BG1

            Rotation BG:    BG2

    mode 2: Raotation BG:   BG2,BG3

    接下来就看看控制这些BG的寄存器:

    1.Text BG

    AgbLib任天堂提供的官方开发包里面是这么写出它的寄存器

    #define REG_BASE                0x04000000          // Registers

    #define REG_BG0CNT      (REG_BASE + 0x8)    // BG 0 Control

    #define REG_BG1CNT      (REG_BASE + 0x8)    // BG 1 Control

    2.Rotation BG

    AgbLib任天堂提供的官方开发包里面是这么写出它的寄存器

    #define REG_BASE                0x04000000          // Registers

    #define REG_BG2CNT      (REG_BASE + 0x8)    // BG 0 Control

    #define REG_BG3CNT      (REG_BASE + 0x8)    // BG 1 Control

    Poriority是指的是显示的优先权.

    0最高,4最低.

    再看看关于BG大小的问题.Screen Size Setting  Text Screen                Rotation/Scaling Screen                       Screen Size  Screen Data   Screen Size  Screen Data  00                   256×256     2 Kbytes      128×128     256 Bytes  01                   512×256     4 Kbytes      256×256     1 Kbyte  10                   256×512     4 Kbytes      512×512     4 Kbytes  11                   512×512     8 Kbytes      1024×1024   16 Kbytes 

    为了显示一张BG.你可以通过AgbLib中提供的bmp2map.exe来把一张bmp文件转换成GBA可以用的BG数据.BG数据分三块.1.Character Data前面已经说了.mode0-2下的GBA屏幕是由图块组成的.这些图块都是8x8大的.这些图块有些人把它叫做Tile,也有些人把它叫做Character(AgbLib里面就是这么叫的).这些数据就是保存这些8x8图块的图象数据的.Character Data最大能有256个.因为16KB = 16384 bytes =256*8*8有些图片转换器不能提示这个最大限制,比如kaleid,即使你的bmp包含超过256个Character,但是它还是依旧给你转换,但是一旦在GBA程序里运行就会出现错误.所以建议使用AgbLib中的bmp2map.exe这个转换工具.2.Map DataMap Data可以说是图块的引索.它是记录Character Data的在屏幕中的排布的数据.3.Palatte Data这个不用说了,就是调色板

    在显示一张BG前,应该把上面的上部分数据传输到GBA内存中.看看下面的代码吧:DmaArrayCopy(3,e1_Character, BG_VRAM,  32);DmaArrayCopy(3,e1_Map,  BG_VRAM+0x8000, 32);DmaArrayCopy(3,e1_Palette,  BG_PLTT+32*2,  32);

    DmaArrayCopy是个数据传输的函数.当然,你也可以使用for(...)一个字节一个字节地写.但是GBA里有3个DMA通道(就是不经过CPU,专门用来传输大量数据的专线),它应该是就像memcpy那样的功能,它的传输速度比memcpy快得多.上面的代码就是将e1图像文件Character,Map,Palette传输到GBA中对应的内存.特别指出的是在使用16色的调色板的时候,你如果用bmp2map.exe来转换.那么就要注意这个程序中-p这个参数.它指出了这个Map用的哪个Paletee.比如我上面的那个e1的BG就是使用了-p2,即使用的第2个调色板.那么它在GBA中对应的内存地址应该是BG_PLTT+32*2(16色的调色板总共占32个字节).你可能要问BG_VRAM,BG_PLTT是什么.#define PLTT                    0x05000000      #define BG_PLTT                 (PLTT +        0x0) #define VRAM                    0x06000000  #define BG_VRAM                 (VRAM +        0x0) 它们其实就是VRAM,Palette在内存中的指定位置.Map Data,Character Data都是放在VRAM中的.在mode0-2中,VRAM被分成许多Block,每个2KB(0x800).Character Data比较大一点,它一个要占8个Block(16KB).Map Data比较小一点,它一个占一个Block.还是看看这张十分有用图片

    Character Base Block的值为0-3 Screen Base Block(就是装Map Data的)的值为0-31 当然,不能让他们重复,否则先写的数据就被覆盖了. 再看看设置BG的寄存器的代码吧: #define TEXTBG_SIZE_256x256  0x0 #define TEXTBG_SIZE_256x512  0x8000 #define TEXTBG_SIZE_512x256  0x4000 #define TEXTBG_SIZE_512x512  0xC000

    #define ROTBG_SIZE_128x128  0x0 #define ROTBG_SIZE_256x256  0x4000 #define ROTBG_SIZE_512x512  0x8000 #define ROTBG_SIZE_1024x1024 0xC000 DmaArrayCopy(3,e1_Character, BG_VRAM,  32); DmaArrayCopy(3,e1_Map,  BG_VRAM+0x8000, 32); DmaArrayCopy(3,e1_Palette,  BG_PLTT+32*2,  32); *(vu16 *)REG_BG0CNT =   BG_COLOR_16 | TEXTBG_SIZE_256x256 | BG_PRIORITY_0   | 16 << BG_SCREEN_BASE_SHIFT | 0 << BG_CHAR_BASE_SHIFT; 你对应前面的图看看. 16是Screen Base Block的值,0是Character Base Block的值. 0x8000就是Base Block 16的对应偏移址,0是Base Block 0的对应偏移址

    GBA硬件上对有很多对BG处理的支持,比如旋转,移动,放大,Alpha,马塞克.

    我们先看看BG的移动. 这同样也是通过对GBA的寄存器的写入来实现的. 在BG2(Rotation BG)中有两个寄存器控制它的位置.REG_BG2X,REG_BG2Y 还是看看它们的定义 #define REG_BASE                0x04000000          // Registers #define REG_BG2X        (REG_BASE + 0x28)   // BG 2 Start X Coordinate #define REG_BG2Y        (REG_BASE + 0x2c)   // BG 2 Start Y Coordinate

    它们是只能写的寄存器,你可不要去读它们,否则读不出来什么有用的数据的. 但是你可以去写它们: s32 x=10<<8; s32 y=10<<8; *(s32 *)REG_BG2X=x; *(s32 *)REG_BG2Y=y; 需要记住的它们都是32位的数据,而且是有符号的.x增大,BG向左移,x减小,BG向右移.y增大,BG向上移,y减小,BG想下移.当x或y每改变256,BG移动1个像素.所以上面的x,y都使用了<<8


    最新回复(0)