• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 60, 文章 - 0, 評(píng)論 - 197, 引用 - 0
            數(shù)據(jù)加載中……

            學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程

            一、目的
               串口通信我們并不陌生,我們經(jīng)常用串口來進(jìn)行數(shù)據(jù)傳輸,可并不清楚它是如何工作
            的。那這一節(jié)我們就來揭開 ARM S3c2410 UART(Universal Asynchronous Receiver and
            Transmitter) 串口通信的神秘面紗。


            二、代碼
               我們先來分析文件 crt0.s
              
               @ 文件 crt0.s
               @ 作用:設(shè)置堆棧指針
               .text
               .global _start
               _start:
                ldr sp, =1024*4
                bl main
               halt_loop:
                b halt_loop
              
               你可能會(huì)有疑問,這個(gè)匯編文件有什么用?呵呵,這是因?yàn)槲覀兊拇谕ㄐ糯a要用 C
            編寫(用匯編可讀性太差了)。可這又和這個(gè) crt0.s 有什么關(guān)系呢?這得從 C 語言程序的
            編譯說起。C 語言程序執(zhí)行的第一條指令并不在 main 函數(shù)里。當(dāng)生成一個(gè) C 語言程序時(shí)
            編譯器總是在我們的代碼前加一段固定的代碼--crt0.o,它是編譯器自帶的一個(gè)文件,用來
            設(shè)置 C 程序的堆棧等,然后調(diào)用 main 函數(shù)。可惜在我們的裸板上它自帶的 crt0.o 的代
            碼是不能運(yùn)行的,我們得自己動(dòng)手寫,這就是為什么要有 crt0.s 這個(gè)文件。稍后你將看到,
            這個(gè) crt0.s 被編譯成我們自己的 crt0.o 文件。


               /* 頭文件 serl.h
                * 作用:定義相關(guān)寄存器、UART 初始化函數(shù)聲明、串口讀寫函數(shù)的聲明
                */
               #ifndef __SERL_H__
               #define __SERL_H__
              
               #define GPHCON  (*(volatile unsigned long *)0x56000070)
              
               /* PORT PULL-UP REGISTER*/
               #define GPHUP   (*(volatile unsigned long *)0x56000078)
              
               /* UART FIFO control register 0*/
               #define UFCON0  (*(volatile unsigned long *)0x50000008)
              
               /* UART line control register 0*/
               #define ULCON0  (*(volatile unsigned long *)0x50000000)
              
               /* UART control register 0*/
               #define UCON0   (*(volatile unsigned long *)0x50000004)
              
               /* UART modem control register 0*/
               #define UMCON0  (*(volatile unsigned long *)0x5000000C)
              
               /* UART baud rate divisor register 0*/
               #define UBRDIV0 (*(volatile unsigned long *)0x50000028)
              
               /* UART TX/RX status register 0*/
               #define UTRSTAT0 (*(volatile unsigned long *)0x50000010)
              
               /* UART transmit buffer register 0*/
               #define UTXH0 (*(volatile unsigned char *)0x50000020)
              
               /* UART receive buffer register 0*/
               #define URXH0 (*(volatile unsigned char *)0x50000024)
              
              
               #define TXD0_READY 0x2
               #define RXD0_READY 0x1
              
               void init_uart();
               void putc(unsigned char ch);
               unsigned char getc();
              
               #endif


               /* 文件 serl.c*/
               #include "serl.h"
              
               void init_uart() {
                 GPHCON |= 0xa0; /* GPH2, GPH3 used as RXD0, TXD0*/
                 GPHUP = 0x0c;   /* GPH2, GPH3 poll-up */
              
                 ULCON0 = 0x03;  /* normal mode, no parity, one stop bit, 8-bit*/
                 UCON0 = 0x05;   /* Loopback mode*/
                 UFCON0 = 0x00;  /* not use FIFO*/
                 UMCON0 = 0x00;  /* disable flow control*/
                 UBRDIV0 = 12;   /* baud rate 57600*/
               }
              
               void putc(unsigned char ch) {
                 while (!(UTRSTAT0 & TXD0_READY));
                 UTXH0 = ch;
               }
              
               unsigned char getc(){
                 while (! (UTRSTAT0 & RXD0_READY));
                 return URXH0;
               }
              

            我們選用最簡單的方法,用 UART0 進(jìn)行實(shí)驗(yàn),用到的寄存器有8個(gè)多,初始化用去5
            個(gè),余下的3個(gè)用于接收、發(fā)送數(shù)據(jù)。初始化設(shè)置的代碼說明如下:
               1. GPHCON 的 GPH2、GPH3用控制接收數(shù)據(jù)寄存器 RXD0 和發(fā)送數(shù)據(jù)寄存器 TXD0
                  手冊(cè)中GPH2、GPH3描述如下:

            GPHCON     Bit         Description
            GPH3 [7:6] 00 = Input    01 = Output   
            10 = RXD0  11 = reserved
            GPH2              [5:4]  00 = Input     01 = Output
            10 = TXD0   11 = Reserved
                                                                                                                          
                  所以
                    GPHCON |= 0xa0
                    GPHUP  |= 0x0c (上拉)

               2. ULCON0 設(shè)置為 0x03, 含義是正常操作模式、無校驗(yàn)、停止位1、8個(gè)數(shù)據(jù)位
               3. UCON0 設(shè)置為 0x05 表示發(fā)送、接收數(shù)據(jù)都使用查詢方式
               4. UFCON0 設(shè)置為 0x00 為不使用 FIFO (每個(gè)UART內(nèi)部都有一個(gè)16字節(jié)的發(fā)送和接收
                  FIFO)
               5. UMCON0 設(shè)置為 0x00 為不使用流控
               6. UBRDIV0 設(shè)置為 12 含義為 波特率設(shè)為 57600, 由下面公式算得:
                    UBRDIVn = (int) (PCLK/bps*16) - 1
                  其中 PCLK = 12MHz


            發(fā)送/接收數(shù)據(jù)的代碼說明如下:
               1. UTRSTA0 (UART TX/RX status register 0 )
                  bit[1]:無數(shù)據(jù)發(fā)送時(shí)自動(dòng)設(shè)為1,我們要用串口發(fā)送數(shù)據(jù)時(shí),先讀此位以判斷是否有
                  數(shù)據(jù)正在發(fā)送。
                  bit[0]:接收緩沖區(qū)是否有數(shù)據(jù),如果有,此位自動(dòng)設(shè)為1,我們需要讀此位來判斷是
                  否接收到了數(shù)據(jù)。
               2. UTXH0: 把要發(fā)送的數(shù)據(jù)寫入此寄存器
               3. URXH0: 讀此寄存器會(huì)得到串口接收到的數(shù)據(jù)


              
               /*
                * 測試代碼 main.c
                * 作用:將從串口接收的數(shù)據(jù)發(fā)回串口
                */
               #include "serl.h"
              
               int main(void) {
                 unsigned char ch;
                 init_uart();
                 while (1) {
                   ch = getc();
                   /* 如果接收到的是回車符就發(fā)送回車和換行符*/
                   if (ch == 0x0d) {
                     putc(0x0d);
                     putc(0x0a);
                   }
                   else {
                     putc(ch);
                   }
                 }
               }
                

               # 文件 Makefile
               # 由代碼文件生成目標(biāo)文件,并連接目標(biāo)文件
               # 最后將連接生成的目標(biāo)文件轉(zhuǎn)換成二進(jìn)制格式
               main:crt0.s serl.c main.c
                    arm-linux-gcc -c -o crt0.o crt0.s
                    arm-linux-gcc -c -o serl.o serl.c
                    arm-linux-gcc -c -o main.o main.c
                    arm-linux-ld  -Ttext 0x00000000 crt0.o serl.o main.o -o main_tmp.o
                    arm-linux-objcopy -O binary -S main_tmp.o main
               clean:
                    rm -f *.o
                    rm -f main


            三、編譯、燒寫、測試
               Make 一下就會(huì)生成我們要的文件 main, 將其通過 JTAG 燒入 Nand Flash。用超級(jí)終
            連接到開發(fā)板,注意波特率設(shè)為 57600,數(shù)據(jù)位 8,無奇偶校正,停止位1,無數(shù)據(jù)流控制。現(xiàn)
            在 Reset 一下的開發(fā)板,在超級(jí)終上輸入一些字符,看到你自己輸入的字符了嗎?呵呵,再試
            一試回車,超級(jí)終端上是不是換到了新的一行? 這就是簡單的串口通信!     

            posted on 2008-01-08 21:47 Normandy 閱讀(7545) 評(píng)論(28)  編輯 收藏 引用 所屬分類: Embeded Area

            評(píng)論

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            好詳細(xì)啊,支持支持!
            2008-01-08 21:54 | hukq

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            不錯(cuò)的文章, 學(xué)習(xí)ing......
            2008-01-09 09:51 | kyle

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            樓主太強(qiáng)了,簡直就是我的偶像!
            2008-01-14 19:09 | hukq

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            好東西。
            if (ch == 0x0d) {
            putc(0x0d);
            putc(0x0a);
            }
            else {
            putc(ch);
            }
            }
            ————————————
            改成
            if (ch == 0x0d)
            putc(0x0a);
            putc(ch);
            是不是也可以?
            2008-01-16 22:40 | qqwe

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            確實(shí)好啊,受教了。
            2008-01-16 22:41 | free

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @qqwe
            可以這樣精簡一下:

            putc(ch);
            if (ch == 0x0d) putc(0x0a);

            0x0d 代表回車
            0x0a 代表換行
            它們的順序應(yīng)該是先回車再換行
            2008-01-17 16:26 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            niu bi niu bi niu bee!!
            2008-01-24 17:19 | niube' son

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            如果要是傳一句“hello,2008”,這怎么傳啊,這里面有數(shù)字和標(biāo)點(diǎn)符號(hào).

            void putc(unsigned char ch) {
            while (!(UTRSTAT0 & TXD0_READY));
            UTXH0 = ch;
            }

            unsigned char getc(){
            while (! (UTRSTAT0 & RXD0_READY));
            return URXH0;
            }
            這兩個(gè)函數(shù)里都是char類型的,有關(guān)系嗎?
            2008-03-19 12:05 | 含著淚的微笑

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @含著淚的微笑
            這兩個(gè)函數(shù)是如下使用的:
            getc() 會(huì)從 UART0 讀取一個(gè)符, 它不管你向 UART0 輸入了什么, 只要有數(shù)據(jù)就讀取, 沒有數(shù)據(jù)就等待。

            putc() 會(huì)向 UART0 輸出一個(gè)字符, 上位機(jī)的軟件(如超級(jí)終端) 可以顯示它輸出的是什么

            我想你的問題是這樣的:
            上位機(jī)向 ARM 板子上的 UART0 發(fā)送了一個(gè)串: "hello,2008", ARM 該如何獲取到這個(gè)串, 是這樣吧?
            可以這樣解決:

            char *buf = (char *)0x30010000;
            char *p = buf;
            char c = getc();
            while (c != 0x0d) {
            *p++ = c;
            c = getc();
            }
            *p = '\0';

            0x0d 指回車, 意思是當(dāng)接收到回車時(shí),停止讀取操作。這時(shí) "hello,2008" 已經(jīng)存入緩沖區(qū) buf 里, 你可以再對(duì)這個(gè) buf 進(jìn)行操作了。
            2008-03-19 15:26 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   


            謝謝你的回復(fù)!!!

            1
            我還想問你一下,如果上位機(jī)不是通過在“超級(jí)終上輸入一些字符”該怎么樣呢??
            比如上位機(jī)自動(dòng)發(fā)送“hello,2008”而不是通過人的輸入,那么下位機(jī)就不能通過
            “c != 0x0d”來判斷輸入的結(jié)束,這時(shí)該怎樣解決呢??
            2我做作業(yè)碰到這樣一個(gè)要求“接收PC發(fā)來的用戶數(shù)據(jù)(變長),并保存到基地址在0x01000000的Flash中,F(xiàn)lash的型號(hào)為 AM29L160B。”
            b"接收PC發(fā)來的用戶數(shù)據(jù)(變長)"就不知道該怎樣判斷了,是不是我要定一個(gè)規(guī)則?比如PC發(fā)來的用戶數(shù)據(jù)要定義一個(gè)結(jié)束標(biāo)志,比如就用"0x0d"來表示數(shù)據(jù)的結(jié)尾,當(dāng)下位機(jī)檢測到這位時(shí),就說明數(shù)據(jù)終止。但這樣也有一個(gè)問題,就是我送的文本中有換行符,豈不是就把一個(gè)完整的文本分割了?
            a.向Flash中寫入數(shù)據(jù),有什么步驟嗎?

            =====
            真是不好意思,來麻煩您!
            再一次致謝!
            2008-03-20 10:29 | 含著淚的微笑

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            1

            "GPHCON 的 GPH2、GPH3用控制接收數(shù)據(jù)寄存器 RXD0 和發(fā)送數(shù)據(jù)寄存器 TXD0"這兩個(gè)寄存器在實(shí)際應(yīng)用中是連線用的嗎?我看有些實(shí)驗(yàn)手冊(cè)中沒有寫用這兩個(gè)寄存器,為何?
            2
            GPHUP |= 0x0c (上拉),上拉有什么作用?是不是往GPHDAT中的位寫1,端口就是1,輸出電流。我做作業(yè)“接收PC命令,實(shí)現(xiàn)開關(guān)門(GPIO0 1:開,0:關(guān))”
            在設(shè)置I/O時(shí)有這方面的疑惑,看處理器手冊(cè)上也沒看明白。

            =======
            麻煩您了!謝謝!
            2008-03-20 11:39 | 含著淚的微笑

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            你好,在你的初使化中,

            UCON0 = 0x05; /* Loopback mode*/
            不知你為何在程序中設(shè)置成這個(gè)模式??

            ------------------

            2008-03-20 12:00 | 含著淚的微笑

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @含著淚的微笑
            歸納了你的問題,做答如下:
            1. 不一定非得用超級(jí)終端, 你也可以自己寫程序。將 PC 的串口和 ARM 開發(fā)板的串口相連后,只要你的程序向 PC 串口發(fā)數(shù)據(jù), ARM 就能通過連線獲得數(shù)據(jù)。
            2. 至于如何標(biāo)識(shí)一個(gè)字串結(jié)束, 這是一個(gè)協(xié)議問題,可指定一個(gè)數(shù),比如 0x55AA,當(dāng)收到的兩個(gè)連續(xù)字符是0x55AA時(shí),則停止接收。
            3. 可先將數(shù)據(jù)存到 SDRAM 緩沖區(qū),再寫到 Nand Flash。至于如何寫 Nand Flash, 可參考 vivi 的源代碼,里面有一個(gè)向 Nand Flash 寫數(shù)據(jù)的方法。
            2008-03-20 12:20 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @含著淚的微笑
            0x05 代表查詢模式,實(shí)現(xiàn)起來最簡單。
            2008-03-20 12:25 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @含著淚的微笑
            1. RXD0 和 TXD0 你并看不到,能連線的只有串口。
            2. 電阻上拉,我的理解是拉高電壓,使電路更穩(wěn)定。
            3. 仔細(xì)閱讀板子附帶的手冊(cè)吧,把這個(gè)例子在你的板子上先跑起來
            2008-03-20 12:37 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            謝謝你啊!!
            2008-03-20 20:22 | 含著淚的微笑

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            Normandy,你好,我這兩天一直在弄串口,無論是用ADS還是ARM-LINUX-GCC編譯的燒進(jìn)NAND FLASH后超級(jí)終端或者DNW都沒有反映。能否將你的這段程序源碼發(fā)給我一份?謝謝。wangfan#gmail.com,#改為@。
            2008-03-23 10:22 | FAN

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @FAN
            你好,已將代碼發(fā)到你郵箱,有問題及時(shí)討論。
            2008-03-24 10:08 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            謝謝,你的代碼是沒問題的,我再看看我的是哪兒出問題了。
            2008-03-24 12:25 | FAN

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            #define TXD0_READY 0x2
            上句中0x2換為0x4在這個(gè)例子中也可以的吧?
            2008-03-24 14:01 | FAN

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            頂上去
            2008-03-24 14:29 | niube' son

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @FAN
            我沒有試過 #define TXD0_READY 0x4 查了一下手冊(cè),上面這樣描述, 對(duì)于 UTRSTA0[2]:

            Set to 1 automatically when the transmit buffer register has
            no valid data to transmit and the transmit shift register is
            empty.
            0 = Not empty
            1 = Transmitter (transmit buffer & shifter register) empty

            我覺得可以,你可以試試,別忘了把結(jié)果告訴我啊 :-)
            2008-03-25 09:34 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @Normandy
            我試過,可以的。
            2008-03-25 13:02 | FAN

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            請(qǐng)問, 你寫過2410的DMA based uart driver嗎? 我的DMA uart發(fā)數(shù)據(jù)沒問題, 但從hyper terminal收數(shù)據(jù)的時(shí)候不能觸發(fā)DMA0 interrupt, 非常疑惑, 不知道從何調(diào)起. 你知道嗎? 謝謝

            2008-08-06 21:33 | 嚴(yán)植

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            @嚴(yán)植
            上面例子是基于查詢而非 DMA ,不過應(yīng)該不難。我想可能是以下的問題:
            (1) 超級(jí)終端收發(fā)數(shù)據(jù)有問題(看看設(shè)置,如波特率等是否正確)
            (2) 板子串口有問題(我的經(jīng)常是這樣, 多插幾次試試)
            (3) UART 設(shè)成 DMA 中斷方式有誤 (請(qǐng)仔細(xì)查看手冊(cè), 確認(rèn)相關(guān)的控制寄存器是否設(shè)置正確)
            2008-08-07 10:35 | Normandy

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            你寫的這種是不像DOS那樣沒有發(fā)送接收的窗口的?
            2009-05-03 09:17 | xiamenxw

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            大哥,

            您好!

            請(qǐng)問一下FS2410的UART口連接筆記本電腦,可以怎么連接?
            2011-12-19 18:08 | 楊軍

            # re: 學(xué)習(xí) ARM 系列 -- FS2410 開發(fā)板上的串口通信編程  回復(fù)  更多評(píng)論   

            你好,ARM網(wǎng)關(guān)串口通信采集根節(jié)點(diǎn)數(shù)據(jù),要求用c寫,需要用到ARM的哪些東西嗎,本人對(duì)ARM一無所知,網(wǎng)上沒有查到這樣的資料,單純c接收串口程序有點(diǎn),可是網(wǎng)關(guān)盒子是ARM的,應(yīng)該怎么做,求大神指教,或者提供相關(guān)資料,不勝感激。
            2014-09-05 11:15 | 王勇
            亚洲AV日韩AV天堂久久| 精品久久一区二区三区| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 91精品国产91久久久久久| 久久久久久免费一区二区三区| 99久久精品国产毛片| 亚洲精品NV久久久久久久久久 | 久久精品国产只有精品2020| 国产精品美女久久久久AV福利| 中文成人无码精品久久久不卡| 午夜精品久久久久久99热| 91精品日韩人妻无码久久不卡| 久久亚洲AV无码精品色午夜麻豆| 国产成人精品久久一区二区三区| 久久人人爽人人爽人人片AV东京热| 久久午夜羞羞影院免费观看| 国产精品美女久久久免费| 亚洲AV无码久久精品狠狠爱浪潮 | 国产伊人久久| 99久久国产热无码精品免费| 久久激情五月丁香伊人| 久久亚洲精品无码AV红樱桃| 久久天天躁狠狠躁夜夜2020| 久久久久久久综合日本亚洲| 无码超乳爆乳中文字幕久久| 国产精品一区二区久久精品涩爱| 国产L精品国产亚洲区久久 | 色偷偷偷久久伊人大杳蕉| 亚洲人成无码www久久久| 国产A级毛片久久久精品毛片| 国内精品人妻无码久久久影院| 99久久国产精品免费一区二区| 久久亚洲国产精品123区| 伊人久久大香线蕉精品| 99久久国语露脸精品国产| 久久国产乱子伦免费精品| 日本人妻丰满熟妇久久久久久| 亚洲熟妇无码另类久久久| 亚洲AV无码1区2区久久| 久久亚洲私人国产精品| 久久九九精品99国产精品|