**********************
* 第二讲 硬件I/O操作 * ********************** 对设备进行访问和控制时需要访问I/O寄存器。硬件抽象层提供了一些宏用于I/O寄存器的读写操作。 之所以不用指针直接操作,目的是为了可移植性。使用宏的原因是避免函数调用引起的性能损失。ecos是一种可移植的嵌入式操作系统,它可以移植到16位、32位、64位的各种处理器平台上。ecos由各种组件构成,根据具体硬件平台的需要可以分别将这些组件加入到系统中来,从而实现各种所需的功能。ecos的这种层次结构的最底层是硬件抽象层(Hardware Abstraction Layer),通常称为HAL。硬件抽象层HAL对处理器结构和系统硬件平台进行抽象,当需要在一个新的目标平台上运行ecos时,只需对底层的硬件抽象层进行修改,便可迅速地将整个ecos系统移植到新的平台上。 常用I/O操作宏: 读寄存器值 从寄存器读数据并将结果存放在存值变量里 HAL_READ_UINT8(寄存器地址,存值变量); HAL_READ_UINT16(寄存器地址,存值变量); HAL_READ_UINT32(寄存器地址,存值变量); 写寄存器 向寄存器写如数值 HAL_WRITE_UINT8(寄存器地址,值); HAL_WRITE_UINT16(寄存器地址,值); HAL_WRITE_UINT32(寄存器地址,值); 这些宏位于头文件cyg/hal/hal_io.h内,感兴趣的读者可以看看这些宏是怎么实现的。 I/O操作宏的定义也是按照ecos的分层原则定义的,HAL表示这是硬件抽象层函数,READ/WRITE表示具体操作,UINTx表示读写宽度,这些宏很容易记忆的,而且见名知意。 寄存器名称定义在cyg/hal/plf_io.h中,我们仍然按照ecos命名规则起名,如下: // GPIO #define LPC2XXX_GPIO_IO0PIN 0xE0028000 #define LPC2XXX_GPIO_IO0SET 0xE0028004 #define LPC2XXX_GPIO_IO0DIR 0xE0028008 #define LPC2XXX_GPIO_IO0CLR 0xE002800C LPC2XXX表示芯片型号,GPIO表示I/O类型,IO-n-ops表示Pn口的操作方式(引脚、方向、设置、清除)。这样的命名比枯燥的数字更容易记忆,不重名也不易错。 下面是控制GPIO让蜂鸣器发声的应用程序,响1秒停1秒,周而复始。线程创建第一讲已经说过,不再重复。 #i nclude <cyg/kernel/kapi.h> #i nclude <cyg/hal/hal_io.h> #i nclude <cyg/hal/plf_io.h> #define STACK_SIZE 4096 #define BEEPCON 0x0000080 char stack[2][STACK_SIZE]; static cyg_thread thread_data[2]; static cyg_handle_t thread_handle[2]; void taska(cyg_addrword_t data) { int message = (int) data; HAL_WRITE_UINT32(LPC2XXX_GPIO_IO0DIR,BEEPCON); for(;;) { HAL_WRITE_UINT32(LPC2XXX_GPIO_IO0SET,BEEPCON); cyg_thread_delay(100); HAL_WRITE_UINT32(LPC2XXX_GPIO_IO0CLR,BEEPCON); cyg_thread_delay(100); } } void test(cyg_addrword_t data) { printf("\n\n\n"); printf("\t *******************************\n"); printf("\t * Hello! The world. *\n"); printf("\t *******************************\n\n\n"); // Create a main thread, so we can run the scheduler and have time 'pass' cyg_thread_create(10, // Priority - just a number taska, // entry 1, // entry parameter "taska", // Name &stack[1], // Stack STACK_SIZE, // Size &thread_handle[1], // Handle &thread_data[1] // Thread data structure ); cyg_thread_resume(thread_handle[1]); // Start it } void cyg_start(void) { // Create a main thread, so we can run the scheduler and have time 'pass' cyg_thread_create(10, // Priority - just a number test, // entry 0, // entry parameter "test", // Name &stack[0], // Stack STACK_SIZE, // Size &thread_handle[0], // Handle &thread_data[0] // Thread data structure ); cyg_thread_resume(thread_handle[0]); // Start it cyg_scheduler_start(); }