CSDN博客

img tangl_99

GBA探索日记(-)

发表于2003/1/13 9:19:00  9862人阅读

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 Data
Map 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 0

相关文章推荐

img
取 消
img