自从学习开发GBA后,便对其开发的容易性,效率为之吸引.
想必大家也是如此.
因为GBA开发多数是使用C语言的,
这里写一点基本的C语言的使用技巧.
希望能增加大家的内功.
1 对内存里的寄存器进行操作.
任何一个GBA开发库的GBA头文件都包含了很多由#define定义的符号.
如 #define VRAM 0x06000000 // VRAM
这里的VRAM是一个宏符号,后面的0x06000000是一个立即数
说明VRAM所代替的是0x06000000.
和变量不同的是,变量会占内存,而宏定义是不占内存的.因为宏只对编译器负责.
由于GBA是使用了统一编地址.所有逻辑功能部件都是直接连接CPU片总线
所以,GBA要对其他部件操作的时候,就如同操作内存一样容易.
下面讲一下如何直接对内存进行操作.
就拿VRAM来说,如果想对这个地址写入一个 8 位无符号数的话,可以如下操作.
*(u8*)VRAM = 0xff; // u8 是 unsigned char
也可以这样写
*(u8*)0x06000000 = 0xff
这样的操作在C语言中是合法的.相当与把一个指针直接指定地址后类型转换成U8然后对内容操作.
其结果是 06000000h 的位置为 0xff
如果你想写入一个u16的数的话,可以这样写
*(u16*)VRAM=0xaaff 或 *(u16*)0x06000000
其结果是 06000000h 为 0xff
06000001h 为 0xaa
以高位存放高地址,低位存放低地址的(大数端)原则存放数据.
当然,地址是固定的,如何使用就要看实际情况了.
上面这样的操作是GBA里最普遍的.由于98%的C语言的书上都没有提到这类用法,
所以在某种程度上,成了新手学习的绊脚石.
2 利用结构型直接对位进行定义名称.
结构型是C语言里极重要和普遍的定义数据的手段.
但是恰恰许多人对结构型了解不是很深入.
结构型的基本定义方式为
struct
{
u8 a,
u8 b;
u8 c;
u8 *d;
}mystr;
如此便定义好了一个结构,其中包括a,b,c三个U8类型的成员.和一个指针d
这个结构的大小是 3 * 8 + 16 = 40 bit. = 5 byte
因为一个指针的大小等于字长,GBA里多数情况下是用THUMB模式开发程序.故为16位.
然后到我们今天的重点.
举个例子,定义一个15bit颜色结构RGB ,其包含 R,G, B.是一共是16位.
格式为 a bbbbb ggggg rrrrr 最高位无用.
许多人会定义成
typedef struct {
U8 R,G,B;
}RGB;
这个结构占3字节,使用的时候用逻辑运算,压成一个16位的点的数据.
其实此操作太麻烦.
先进的定义方式是:
typedef struct {
u16 r : 5 ;
u16 g : 5 ;
u16 b : 5 ;
u16 dummy : 1 ;
}RGB;
此结构大小为 16bit 符合RGB的规则.
关键在于在成员变量的后面加上 冒号 和 定义的位数.就完成了直接给位进行定义名称.
其中 严格按照从低位到高位的原则,先定义这为低位.后面定义的会接在前面一个后面的位定义.
由于RGB定义完了,使用了15位,有最高位空闲,.所以定义个1位的DUMMY,防止浪费.
使用的时候和一般的结构型一样使用.不过如果数值超出范围的话,超出的部分无效.
如果这样定义
typedef struct
{
u8 a:5 ;
u8 b;
u8 c:4;
}ABC
那么这个结构仍然为3BYTE. 因为成员b没有说明是跟在a后面定义,而是另外重新定义一个成员.故
无法连在a后.
也就是说 a 的高3位就浪费了.
3 实例
就拿GBA里一个十分重要的寄存器DISPCNT来说
位于0x4000000 大小为16bit
头文件里定义为
#define DISPCNT 0x4000000
具体内容为
F E D C B A 9 8 7 6 5 4 3 2 1 0
W V U S L K J I F D B A C M M M
0-2 (M) = BG模式 0 ~ 5
3 (C) = Game Boy Color 模式
4 (A) = BG反转
5 (B) = hblnk
6 (D) = 1D 方式还是2D方式
7 (F) = MODE4中使用.用于检测是哪个FRAME有效.
8 (I) = BG0. 允许显示
9 (J) = BG1. 允许显示
A (K) = BG2. 允许显示
B (L) = BG3. 允许显示
C (S) = OAM 允许显示
E (V) = Window允许
F (W) = Sprite Windows允许
按照上面说的,可以如此定义结构.
typedef struct DispCnt{
u16 BgMode:3; // BG Mode Select
u16 CgbMode:1; // CGB Mode Select
u16 Bmp_FrameNo:1; // Bitmap Mode Display Frame Select
u16 Obj_H_Off:1; // OBJ Processing in H Blank OFF
u16 ObjCharMapType:1; // OBJ Character Data Mapping Type
u16 Lcdc_Off:1; // LCDC OFF
u16 Bg0_On:1; // BG 0 ON
u16 Bg1_On:1; // BG 1 ON
u16 Bg2_On:1; // BG 2 ON
u16 Bg3_On:1; // BG 3 ON
u16 Obj_On:1; // OBJ ON
u16 Win0_On:1; // Window 0 ON
u16 Win1_On:1; // Window 1 ON
u16 ObjWin_On:1; // OBJ Window ON
} DispCnt;
对其进行操作:
((DispCnt*) DISPCNT)-成员变量.
把DISPCNT的地址转换成DispCnt类型然后指针化,把这个指针操作其内部的成员.
这样够方便了把.不用理会那些麻烦的标志,和逻辑操作.
直接赋值就可以了.
(此种方法在99%的C语言书里也没提到过. ……..心寒)
由于GBA里类似的寄存器实在太多,请切记,上述方法使用不要过度.
否则一大堆指针,会占太多的内存.使可用的内存减少,造成资源短缺.
请一定要注意.
写完后,有点累了.希望能帮助大家进入GBA的开发圈子,提高自己的C语言内功.