**********************
* 第二讲 硬件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();      
}