通常我們認(rèn)為以計(jì)算機(jī)CPU為核心,其外部的所有的設(shè)備都可以稱為是外部輸入輸出設(shè)備。例如計(jì)算機(jī)中的顯示器就是一個(gè)輸出設(shè)備,它的作用是將一些數(shù)字信號(hào)轉(zhuǎn)化為圖形信號(hào)顯示在電子屏幕上,其數(shù)據(jù)是由內(nèi)向外流動(dòng),因此我們稱顯示器為輸出設(shè)備。
又如鍵盤和鼠標(biāo)是輸入設(shè)備,它們的作用是將使用者的命令信號(hào)轉(zhuǎn)化為數(shù)字信號(hào)傳遞給計(jì)算機(jī)的CPU,其數(shù)據(jù)是由外向內(nèi)流動(dòng),因此我們稱鍵盤和鼠標(biāo)為輸入設(shè)備。又如,打印機(jī)是一個(gè)輸出設(shè)備,游戲手柄和攝像頭是輸入設(shè)備。如下圖:
通常的,我們將鍵盤稱為計(jì)算機(jī)的標(biāo)準(zhǔn)輸入設(shè)備,將顯示器稱為計(jì)算機(jī)的標(biāo)準(zhǔn)輸出設(shè)備。這也是Unix和類Unix系統(tǒng)中一直延用的名稱。我們?cè)谧鎏摂M文件系統(tǒng)時(shí)將每一個(gè)進(jìn)程中都分配了一個(gè)文件描述結(jié)構(gòu)體數(shù)組:
//進(jìn)程控制塊Process Control Block
typedef struct pcb_s
{
//進(jìn)程棧頂?shù)刂?/span>
void *p_stack;
//棧內(nèi)存地址,釋放、統(tǒng)計(jì)內(nèi)存時(shí)使用
void *p_stack_mem;
//棧內(nèi)在大小
uint32_t stack_size;
//優(yōu)先級(jí)由高0到低32
uint8_t prio;
//任務(wù)狀態(tài)
uint8_t status;
//任務(wù)休眠ticks
uint32_t sleep_tick;
//任務(wù)入口函數(shù)
void (*task_entry)(void *);
//任務(wù)函數(shù)參數(shù)
void *task_arg;
//進(jìn)程的文件描位圖,1表示空閑,0表示使用
uint32_t f_use_map;
//進(jìn)程的文件描述結(jié)構(gòu)體數(shù)組
vfs_node_s *fnodes[FNODE_SIZE];
} pcb_s;
這個(gè)fnodes[FNODE_SIZE]文件描述數(shù)組記錄了進(jìn)程所打開(kāi)每一個(gè)設(shè)備文件的地址。這個(gè)數(shù)組的下標(biāo)就是我們使用open()函數(shù)所返回的值,也就是我們通常所說(shuō)的文件描述符。文件描述符為int類型,范圍通常是0~FNODE_SIZE。當(dāng)文件描述符小于0時(shí)表示打開(kāi)設(shè)備文件失敗。幾乎所有的類Unix系統(tǒng)中都使用了標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤這3個(gè)IO設(shè)備。
所以,操作系統(tǒng)為每一個(gè)進(jìn)程分配文件描述符時(shí),會(huì)默認(rèn)將0、1、2分別用于表示標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤這3個(gè)設(shè)備。所以通常情況下我們使用open()函數(shù)來(lái)打開(kāi)一個(gè)設(shè)備文件時(shí), 返回的文件描述大多數(shù)是以3開(kāi)始的,在同一個(gè)進(jìn)程中同時(shí)打開(kāi)多個(gè)設(shè)備文件,其文件描述符通常會(huì)是3、4、5、6、7、8……等。
實(shí)際上,我們?cè)?a href="http://www.www27dydycom.cn/soft/data/21-22/" target="_blank">嵌入式領(lǐng)域里的處理器性能有限,外設(shè)的各類和功能多種多樣,并非像個(gè)人電腦或是網(wǎng)絡(luò)服務(wù)器一樣通用和統(tǒng)一。在單片機(jī)領(lǐng)域中我們所使用的操作系統(tǒng)并非一定要遵循Unix標(biāo)準(zhǔn)或習(xí)慣,但為了學(xué)習(xí)其優(yōu)秀的設(shè)計(jì)理念,我們可以將我們的嵌入式操作系統(tǒng)中實(shí)現(xiàn)標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤這3個(gè)設(shè)備。
在Cortex-M3處理器中我們可以使用串口設(shè)備作為操作系統(tǒng)中的標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出,而從本質(zhì)上講標(biāo)準(zhǔn)錯(cuò)誤這個(gè)設(shè)備的功能跟標(biāo)準(zhǔn)輸出是一樣的,只不過(guò)其顯示的內(nèi)容都是程序錯(cuò)誤,我們不單獨(dú)來(lái)實(shí)現(xiàn)標(biāo)準(zhǔn)錯(cuò)誤設(shè)備,而只來(lái)完成標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出這兩個(gè)設(shè)備。例如串口1的初始化、讀、寫程序如下:
void serial1_init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);
USART_InitStructure.USART_BaudRate = SERIAL_BAUTRATE;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
USART1- >CR1 |= (USART_CR1_RE | USART_CR1_TE);
USART_Init(USART1, &USART_InitStructure);
USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);
USART_Cmd(USART1, ENABLE);
}
void serial1_write(uint8_t data)
{
uint8_t next_head = serial1_tx_buffer_head + 1;
if (next_head >= TX_RING_BUFFER1)
{
next_head = 0;
}
while (next_head == serial1_tx_buffer_tail);
serial1_tx_buffer[serial1_tx_buffer_head] = data;
serial1_tx_buffer_head = next_head;
USART1- >CR1 |= USART_FLAG_TXE;
}
int serial1_read(uint8_t *ch)
{
uint8_t tail = serial1_rx_buffer_tail;
if (serial1_rx_buffer_head == tail)
{
return 0;
}
else
{
uint8_t data = serial1_rx_buffer[tail];
tail++;
if (tail >= RX_RING_BUFFER1)
{
tail = 0;
}
serial1_rx_buffer_tail = tail;
*ch = data;
return 1;
}
}
void storeHandleDataIn(uint8_t data)
{
uint8_t next_head;
next_head = serial1_rx_buffer_head + 1;
if (next_head >= RX_RING_BUFFER1)
{
next_head = 0;
}
if (next_head != serial1_rx_buffer_tail)
{
serial1_rx_buffer[serial1_rx_buffer_head] = data;
serial1_rx_buffer_head = next_head;
}
else
{
next_head++;
next_head--;
}
}
之后我們就可以編寫一個(gè)/dev/ttyS1設(shè)備文件用于串口1的驅(qū)動(dòng):
int ttyS1_open(struct file *fs)
{
return 0;
}
int ttyS1_close(struct file *fs)
{
return 0;
}
size_t ttyS1_read(struct file *fs, void *buff, size_t size)
{
uint8_t *p = (uint8_t *)buff;
size_t read_len = 0;
for (int i = 0; i < size; i++)
{
if (!serial1_read(&p[read_len]))
{
return read_len;
}
read_len++;
}
return size;
}
size_t ttyS1_write(struct file *fs, const void *buff, size_t size)
{
uint8_t *p = (uint8_t *)buff;
for (int i = 0; i < size; i++)
{
serial1_write(p[i]);
}
return size;
}
void ttyS1_init(void)
{
file_operations_s ops = {0};
ops.open = ttyS1_open;
ops.close = ttyS1_close;
ops.write = ttyS1_write;
ops.read = ttyS1_read;
ops.ioctl = NULL;
fs_register_dev("/dev/ttyS1", ops);
}
-
處理器
+關(guān)注
關(guān)注
68文章
19896瀏覽量
235345 -
計(jì)算機(jī)
+關(guān)注
關(guān)注
19文章
7663瀏覽量
90833 -
Unix系統(tǒng)
+關(guān)注
關(guān)注
0文章
15瀏覽量
9819 -
Cortex-M3
+關(guān)注
關(guān)注
9文章
276瀏覽量
60271 -
串口輸出
+關(guān)注
關(guān)注
0文章
17瀏覽量
7678
發(fā)布評(píng)論請(qǐng)先 登錄
力天手把手教你學(xué)單片機(jī)視頻全集下載
小七免殺論壇vip 2013源碼免殺培訓(xùn)課程
【連載貼】【NetRotuer之像學(xué)單片機(jī)一樣學(xué)linux筆記】一、目錄篇
【IMX6UL開(kāi)發(fā)板試用體驗(yàn)】- 項(xiàng)目前篇 - USB 設(shè)備驅(qū)動(dòng)實(shí)戰(zhàn)篇
【電子書】《HELLO FPGA》- 項(xiàng)目實(shí)戰(zhàn)篇
STM32MP1系列教程之Cortex-M4開(kāi)發(fā)篇
Linux和RTOS的時(shí)鐘和定時(shí)器怎么使用
觸摸按鍵控制LED學(xué)習(xí)筆記
筆記本無(wú)線上網(wǎng)之實(shí)戰(zhàn)篇
HELLO FPGA項(xiàng)目實(shí)戰(zhàn)篇的PDF電子書免費(fèi)下載

劉潤(rùn)5分鐘商學(xué)院之實(shí)戰(zhàn)篇電子版下載
【單片機(jī)】實(shí)戰(zhàn)篇:Keil+Proteus數(shù)碼管計(jì)數(shù)99

INTEL FPGA學(xué)習(xí)筆記

評(píng)論