• <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, 評論 - 197, 引用 - 0
            數(shù)據(jù)加載中……

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

            一、目的
               串口通信我們并不陌生,我們經(jīng)常用串口來進行數(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
              
               你可能會有疑問,這個匯編文件有什么用?呵呵,這是因為我們的串口通信代碼要用 C
            編寫(用匯編可讀性太差了)。可這又和這個 crt0.s 有什么關(guān)系呢?這得從 C 語言程序的
            編譯說起。C 語言程序執(zhí)行的第一條指令并不在 main 函數(shù)里。當(dāng)生成一個 C 語言程序時
            編譯器總是在我們的代碼前加一段固定的代碼--crt0.o,它是編譯器自帶的一個文件,用來
            設(shè)置 C 程序的堆棧等,然后調(diào)用 main 函數(shù)。可惜在我們的裸板上它自帶的 crt0.o 的代
            碼是不能運行的,我們得自己動手寫,這就是為什么要有 crt0.s 這個文件。稍后你將看到,
            這個 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 進行實驗,用到的寄存器有8個多,初始化用去5
            個,余下的3個用于接收、發(fā)送數(shù)據(jù)。初始化設(shè)置的代碼說明如下:
               1. GPHCON 的 GPH2、GPH3用控制接收數(shù)據(jù)寄存器 RXD0 和發(fā)送數(shù)據(jù)寄存器 TXD0
                  手冊中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, 含義是正常操作模式、無校驗、停止位1、8個數(shù)據(jù)位
               3. UCON0 設(shè)置為 0x05 表示發(fā)送、接收數(shù)據(jù)都使用查詢方式
               4. UFCON0 設(shè)置為 0x00 為不使用 FIFO (每個UART內(nèi)部都有一個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è)為1,我們要用串口發(fā)送數(shù)據(jù)時,先讀此位以判斷是否有
                  數(shù)據(jù)正在發(fā)送。
                  bit[0]:接收緩沖區(qū)是否有數(shù)據(jù),如果有,此位自動設(shè)為1,我們需要讀此位來判斷是
                  否接收到了數(shù)據(jù)。
               2. UTXH0: 把要發(fā)送的數(shù)據(jù)寫入此寄存器
               3. URXH0: 讀此寄存器會得到串口接收到的數(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)換成二進制格式
               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 一下就會生成我們要的文件 main, 將其通過 JTAG 燒入 Nand Flash。用超級終
            連接到開發(fā)板,注意波特率設(shè)為 57600,數(shù)據(jù)位 8,無奇偶校正,停止位1,無數(shù)據(jù)流控制。現(xiàn)
            在 Reset 一下的開發(fā)板,在超級終上輸入一些字符,看到你自己輸入的字符了嗎?呵呵,再試
            一試回車,超級終端上是不是換到了新的一行? 這就是簡單的串口通信!     

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

            評論

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

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

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

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

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

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

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

            好東西。
            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ù)  更多評論   

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

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

            @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ù)  更多評論   

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

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

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

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

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

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

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

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

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

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

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

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


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

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

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

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

            1

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

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

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

            你好,在你的初使化中,

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

            @FAN
            我沒有試過 #define TXD0_READY 0x4 查了一下手冊,上面這樣描述, 對于 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ù)  更多評論   

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

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

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

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

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

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

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

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

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

            大哥,

            您好!

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

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

            你好,ARM網(wǎng)關(guān)串口通信采集根節(jié)點數(shù)據(jù),要求用c寫,需要用到ARM的哪些東西嗎,本人對ARM一無所知,網(wǎng)上沒有查到這樣的資料,單純c接收串口程序有點,可是網(wǎng)關(guān)盒子是ARM的,應(yīng)該怎么做,求大神指教,或者提供相關(guān)資料,不勝感激。
            2014-09-05 11:15 | 王勇
            国内精品久久久久久久影视麻豆| 国产精品嫩草影院久久| 亚洲精品无码久久毛片| 久久精品国产99久久久古代| 久久99精品久久久久久久不卡| 亚洲伊人久久大香线蕉苏妲己| 久久久久无码精品| 久久丫精品国产亚洲av不卡| 国产毛片久久久久久国产毛片| 97精品依人久久久大香线蕉97 | 国产99久久久国产精品小说| 亚洲AV无码久久精品蜜桃| 久久人人爽人人爽人人片AV麻豆 | 99久久精品这里只有精品| 一本久久a久久精品亚洲| 伊人久久免费视频| 色综合久久久久久久久五月| 99久久精品国产一区二区蜜芽 | 99精品伊人久久久大香线蕉| 日韩久久久久久中文人妻 | 狠狠色丁香久久综合婷婷| 久久亚洲sm情趣捆绑调教| 国产一区二区精品久久岳| 国内精品九九久久久精品| 久久精品国产亚洲av麻豆图片| 久久久WWW成人免费精品| 亚洲精品国产成人99久久| 久久99精品久久久久子伦| 久久人人爽人人爽人人片AV不| 国产毛片久久久久久国产毛片| 久久免费视频网站| 精品无码久久久久久尤物| 久久亚洲私人国产精品vA| 久久精品国产亚洲av麻豆蜜芽| 久久笫一福利免费导航 | 97久久超碰国产精品2021| 久久久这里有精品| 伊色综合久久之综合久久| 2021久久精品免费观看| 久久精品成人欧美大片| 亚洲AV日韩精品久久久久久久|