• <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;
               }
              

            我們選用最簡(jiǎn)單的方法,用 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ù)


              
               /*
                * 測(cè)試代碼 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


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

            posted on 2008-01-08 21:47 Normandy 閱讀(7534) 評(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)了,簡(jiǎn)直就是我的偶像!
            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
            可以這樣精簡(jiǎn)一下:

            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ù)!?。?br>
            1
            我還想問你一下,如果上位機(jī)不是通過在“超級(jí)終上輸入一些字符”該怎么樣呢??
            比如上位機(jī)自動(dòng)發(fā)送“hello,2008”而不是通過人的輸入,那么下位機(jī)就不能通過
            “c != 0x0d”來判斷輸入的結(jié)束,這時(shí)該怎樣解決呢??
            2我做作業(yè)碰到這樣一個(gè)要求“接收PC發(fā)來的用戶數(shù)據(jù)(變長(zhǎng)),并保存到基地址在0x01000000的Flash中,F(xiàn)lash的型號(hào)為 AM29L160B?!?br>b"接收PC發(fā)來的用戶數(shù)據(jù)(變長(zhǎng))"就不知道該怎樣判斷了,是不是我要定一個(gè)規(guī)則?比如PC發(fā)來的用戶數(shù)據(jù)要定義一個(gè)結(jié)束標(biāo)志,比如就用"0x0d"來表示數(shù)據(jù)的結(jié)尾,當(dāng)下位機(jī)檢測(cè)到這位時(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)起來最簡(jiǎ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高清漫画| 麻豆精品久久精品色综合| 久久久久久久人妻无码中文字幕爆| 中文字幕无码精品亚洲资源网久久| 久久66热人妻偷产精品9| 国产成人久久久精品二区三区| 久久精品人人做人人妻人人玩| 日韩一区二区久久久久久| 要久久爱在线免费观看| a高清免费毛片久久| 久久综合鬼色88久久精品综合自在自线噜噜 | 久久国产精品偷99| 亚洲国产精品久久久天堂| 韩国三级中文字幕hd久久精品 | 久久夜色精品国产噜噜亚洲a| 久久久久久亚洲精品成人| 久久人妻少妇嫩草AV无码蜜桃| 99久久er这里只有精品18| 久久久久国产精品嫩草影院| 91久久精品电影| 久久亚洲精品人成综合网| 午夜精品久久久久久久无码| 93精91精品国产综合久久香蕉| 国产精品久久午夜夜伦鲁鲁| 久久无码专区国产精品发布| 久久夜色精品国产www| 久久精品国产精品亚洲人人 | 精品国产乱码久久久久软件| 国产成人久久久精品二区三区 | 热久久最新网站获取| 久久精品国产精品亚洲人人| 国产精品美女久久久久AV福利| 成人久久久观看免费毛片| 成人午夜精品无码区久久| 久久综合九色综合网站| 精品久久久久久久久免费影院| 欧美性大战久久久久久| 欧美亚洲日本久久精品| 性做久久久久久免费观看| 久久天天躁狠狠躁夜夜av浪潮 | 国产高潮国产高潮久久久|