久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2811|回復: 4
打印 上一主題 下一主題
收起左側

stc15單片機串口不能發送

[復制鏈接]
跳轉到指定樓層
樓主
ID:691449 發表于 2020-3-14 21:22 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
這幾天用單片機做了一個東西,用到了adc,uart串口,模擬i2c(oled用)。adc用了兩個通道,uart用的串口1,就是下載程序用的P3.0和P3.1。模擬i2c中的scl用p3.6,sda用p3.7。我原來做了很多串口的東西,感覺串口已經不難了。我就先把 adc 和 i2c的oled12864 做好了,adc能正常轉換,oled也能顯示漢字。我最后加上了串口,結果發現反而是我覺得簡單的串口出了問題
串口是這樣的,單片機串口發送一字節,電腦上的串口助手能收到,但是單片機發送完馬上自動復位,就是崩潰了。
我認為可能是這個單片機的串口出了問題,就重新往里燒寫了一個最簡單的串口通信程序,結果簡單的程序可以發送,就是說還是這個程序出了問題。
我又覺得是adc或i2c干擾了串口,于是把adc和i2c都關掉了,又換了引腳位置,結果還是不行。

我為了測試,在程序的開頭發送一字節串口,因為出了問題,所以單片機在一直復位,導致oled不顯示,但是串口初始化不會崩潰,往sbuf里寫數就會出問題。而在發送之前的oled初始化能正常運行。
我現在根本不知道問題出在哪,這又是一個我認為最簡單的串口發送,所以我也找不到原因。

單片機用的stc15w4k32s4,有多串口,我只用了p3.0和p3.1。也不知道是不是這一點出了問題。
放上一部分代碼,oled沒問題。

void main()
{
uchar i,x,y;
ch=0x00;
IOMode00();
OLED_Init(0x20,0x00);            //oled12864初始化,可以正常運行
InitADC(0x00,0x0C);               //adc初始化,可以運行
serialinit(1,11059200,9600);    //串口初始化

SBUF=0x20;                          //=========串口發送,有問題==========
while(!TI);
    TI=0;

OLED16dot(0,2,0,xytab);
OLED16dot(0,4,1,xytab);
for(i=0;i<5;i++)
{
OLED16dot(16+i*16,0,i,titletab);
}
for(i=0;i<5;i++)
{
OLED16dot(16+i*16,6,i,keytab);
}
//OLED816dot(16,2,4,numbertab);
//OLED816dot(16,4,5,numbertab);
while(1){
Delay30ms();
x=GetADC(2);
y=GetADC(3);

dispnum(x,2);
dispnum(y,4);
if (P14){
        OLED16dot(48,6,2,keytab);
        ch&=0x0F;}
        else {
        OLED16dot(48,6,5,keytab);
        ch|=0x10;}

selectxy(x,y);

}
}


關于oled的放上來其實都沒有什么用。
放一下串口和adc


void serialinit(int port,ULONG32 FOSC,UINT16 BAUD)        //參數:串口號,晶振頻率(頻率乘1000000=10^6),波特率
                                                                                                        //(最高波特率65535,串口助手65535之內最高可選57600)
{                                                                                                       
        switch(port)
        {
        case 2:   
                P_SW1 &= 0x7F;                                //將P_SW1前兩位設置成01  (P3.6/RxD_2, P3.7/TxD_2)
                break;
        case 3:
                P_SW1 &= 0xBF;                                //將P_SW1前兩位設置成10  (P1.6/RxD_3, P1.7/TxD_3)
                break;
        case 1:
        default:
                P_SW1 &= 0x3F;                                //將P_SW1前兩位設置成00         (P3.0/RxD, P3.1/TxD)
                break;
        }
    SCON = 0x50;                //8位可變波特率
    T2L = (65536 - (FOSC/4/BAUD));   //設置波特率重裝值
    T2H = (65536 - (FOSC/4/BAUD))>>8;
    AUXR = 0x15;                //T2為1T模式, 并啟動定時器2
    //AUXR |= 0x01;               //選擇定時器2為串口1的波特率發生器
    ES = 1;                     //使能串口1中斷
    EA = 1;
}

void sendbyte(UCHAR8 PrintByte)
{
        //_push_(ACC);
    //ACC = PrintByte;                  //獲取校驗位P (PSW.0)
        while(!TI);
    TI=0;
    SBUF = PrintByte;                 //寫數據到UART數據寄存器   
        //_pop_(ACC);                 
}


串口初始化前面的switch都是判斷串口號的,只用到了串口1,不用看
adc程序:
//============初始化ADC============

void InitADC(uchar Speed,uchar kaiguan)          //速度從低到高:  0x00                0x20                0x40                0x60
                                                                                          //                        540個時鐘        360個時鐘         180個時鐘         90個時鐘
{
    P1ASF = kaiguan;                   //設置P1口為AD口
    ADC_RES = 0;                    //清除結果寄存器
    //ADC_CONTR = ADC_POWER | ADC_SPEEDLL;
        //CLK_DIV |= 0x20;
        ADC_CONTR=0x80|Speed;                        //簡化上面一行
    Delay1ms();                       //ADC上電并延時
}

uchar GetADC(uchar ch)
{
        //uint adcres = 0;
    //ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ch | ADC_START;
        //P1ASF = 0xff;
        ADC_RES=0;
        ADC_CONTR=0x88|ch;                                //簡化上面一行
    _nop_();                        //等待4個NOP
    _nop_();
    _nop_();
    _nop_();
    while (!(ADC_CONTR & 0x10));//等待ADC轉換完成
    //ADC_CONTR &= ~ADC_FLAG;         //Close ADC
        ADC_CONTR=0x80;
        //P1ASF = 0x00;         
    return ADC_RES;                 //返回ADC結果
}

adc只用了8位。

誰能幫我看看啊
分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:213173 發表于 2020-3-15 09:15 | 只看該作者
樓主說“串口初始化前面的switch都是判斷串口號的,只用到了串口1,不用看”,可是switch里分支寫法有誤。只有port不是2或3,由于case 1:后面沒有返回語句break;,port=1或其它數都會執行到default:P_SW1 &= 0x3F;break;。這雖然不至于使P_SW1誤置,但寫法不規范。更嚴重的錯誤在串口發送程序:



void sendbyte(UCHAR8 PrintByte)
{
        //_push_(ACC);
        //ACC = PrintByte;//獲取校驗位P (PSW.0)
        while(!TI);
        TI=0;
        SBUF = PrintByte; //寫數據到UART數據寄存器   
        //_pop_(ACC);                 
}

因為while(!TI);始終為真,程序就死在這。正確寫法:
void sendbyte(UCHAR8 PrintByte)
{
    SBUF = PrintByte; //寫數據到UART數據寄存器   
    while(!TI);//等待發送中斷請求標志位為1
    TI=0;//發送中斷請求標志位清0
}


回復

使用道具 舉報

板凳
ID:691449 發表于 2020-3-15 12:58 | 只看該作者
wulin 發表于 2020-3-15 09:15
樓主說“串口初始化前面的switch都是判斷串口號的,只用到了串口1,不用看”,可是switch里分支寫法有誤。 ...

你好。
但是我在main的下面寫了
SBUF=0x20;                          //=========串口發送,有問題==========
while(!TI);
    TI=0;
這個地方也會出問題
回復

使用道具 舉報

地板
ID:691449 發表于 2020-3-16 20:42 | 只看該作者
現在還是不行,我這個問題本來以為很簡單
回復

使用道具 舉報

5#
ID:213173 發表于 2020-3-16 22:20 | 只看該作者
xianming 發表于 2020-3-16 20:42
現在還是不行,我這個問題本來以為很簡單

其實串口通訊確實是挺簡單的,樓主莫不是開了其他中斷程序干擾了串口通訊。
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 天堂在线www | 午夜无码国产理论在线 | 99re66在线观看精品热 | 免费高潮视频95在线观看网站 | 国产精品激情小视频 | 欧美黑人一区 | 免费在线看黄 | 91久久伊人 | 成人久草 | 欧美日韩综合 | 日韩中文在线视频 | 成人免费在线电影 | 欧美日日 | 精品国产一区二区三区av片 | 日日碰狠狠躁久久躁96avv | 精品三区 | 日韩在线小视频 | 欧美精品一区三区 | yiren22综合网成人 | xxxxx黄色片 欧美一区免费 | 国产综合av | 久久国产精品免费一区二区三区 | 日韩三级在线 | 五月激情婷婷网 | 成年人视频免费在线观看 | 污片在线免费观看 | 超碰在线人人 | 久久福利电影 | 夜夜艹天天干 | 在线精品国产 | 鲁视频| 国产视频精品区 | 日韩一区在线播放 | xxxxxx国产 | 欧美国产激情二区三区 | 日本在线小视频 | 亚洲国产欧美一区二区三区久久 | 欧美日韩久 | 国产三级一区二区三区 | 激情欧美日韩一区二区 | 精品一区二区在线观看 |