通过PIC16F877A的UART发送10位数据

时间:2015-05-26 17:53:51

标签: c

我是微控制器技术的先驱。我想传输从模数转换得到的10位输出,但只能通过UART发送8位。如何发送10位?

请帮我编写C代码来解决这个问题。到目前为止,我的代码如下。使用的编译器是XC8。

#pragma config PWRTE = OFF      // Power-up Timer Enable bit (PWRT disabled)
#pragma config BOREN = ON       // Brown-out Reset Enable bit (BOR enabled)
#pragma config LVP = OFF        // Low-Voltage (Single-Supply) In-Circuit Serial Programming Enable bit (RB3 is digital I/O, HV on MCLR must be used for programming)
#pragma config CPD = OFF        // Data EEPROM Memory Code Protection bit (Data EEPROM code protection off)
#pragma config WRT = OFF        // Flash Program Memory Write Enable bits (Write protection off; all program memory may be written to by EECON control)
#pragma config CP = OFF         // Flash Program Memory Code Protection bit (Code protection off)
#define _XTAL_FREQ 4000000

#include <stdio.h>
#include <stdlib.h>

#include <htc.h>

void uart_init(void);
void TX(unsigned char TX_BYTE);
void configure_pins();
unsigned char read_input(unsigned char channel);

void main()
{
    __delay_ms(2);
    while (1) {
        TRISB = 0;  //configuring portB as output
        TRISC = 0;
        TRISA = 1;

        configure_pins();                   //calling methods
        unsigned char x = read_input(0);
        uart_initialize();
        assign_data_to_tx_pin(x);
    }
}

void configure_pins(){
    ADCON1 = 0b10000000;             //The result is right justified
}

unsigned char read_input(unsigned char channel){  // converting the Analog input to digital
    ADCON0=0b00000000;
    CHS0=0;                  // AN0 is selected
    CHS1=0;                  // "
    CHS2=0;                  // "

    ADON = 1;
    GO_DONE = 1;

    while (GO_DONE);
    ADON = 0;
    return ((ADRESH >> 2) + ADRESL);      // return the result of conversion
}

void uart_initialize(void)            // initializing the UART for data transmission
{

    TRISC = 0;           //configuring portC as output

    TXSTA = 0b100000000;
    TXEN = 1;            //enable transmission mode
    SPEN = 1;            //enable UART
    BRGH = 0;            //enable low baud
    SPBRG = 6;           //set baud rate as 9600
    SYNC = 0;            //enable asynchronous transmission

    RCIE = 1;
    GIE = 1;
    PEIE = 1;
}

void assign_data_to_tx_pin(unsigned char converted_data) {  // assigning the data to the Tx pin for transmission
    while(!TRMT) {
        unsigned char a = converted_data;

        TXREG = a; 
        TXREG = a >> 2; 
        PORTCbits.RC6 = TXREG;

        __delay_ms(100); // Delay
    }
}

2 个答案:

答案 0 :(得分:1)

典型的UART每次传输不允许超过8位数据。有些允许9.可以在选择的UARTS上使用9位发送10位并控制奇偶校验,但这种情况很少见。

而是建议将数据作为2传输发送,并留出一个位以表示发送了哪一半。

Send_ADC(void){
  ADCON0=0b00000000;
  CHS0=0;                  // AN0 is selected
  CHS1=0;                  // "
  CHS2=0;                  // "
  ADON = 1;
  GO_DONE = 1;
  while (GO_DONE);
  ADON = 0;
  unsigned adc10 = ((ADRESH >> 2) + ADRESL); 
  assign_data_to_tx_pin((adc10 % 32) * 2 + 0);   // 00lllll0
  assign_data_to_tx_pin((adc10 / 32) * 2 + 1);   // 00hhhhh1
}

在接收方,确保接收的字节顺序正确。这将以正确的顺序重新构建接收的数据,即使接收没有开始同步或者通信中丢失了一个字节。

// return 0: success, else 1
int ReadSerialADC(unsigned *data) {
  unsigned adc;
  unsigned low = read_from_comport();
  if (low %2) return 1;
  low /= 2;
  unsigned high = read_from_comport();
  if (high %2 == 0) return 1;
  high /= 2;
  *data = high * 32 + low;
  return 0;
}

答案 1 :(得分:0)

你正在读这样的10位ADC结果

return ((ADRESH>>2)+ADRESL);

但该函数返回unsigned char,它应该是unsigned int

unsigned int read_input(unsigned char channel)

并且调用函数也用

丢弃了两位
unsigned char x=read_input(0);

应该是

unsigned int x=read_input(0);

将(读取)16位变量中的10位值读取后,您现在必须将其传输到串行端口。让我们先发送最重要的8位来做到这一点。

TX (x >> 8);
TX (x & 0xFF);

然后在接收器端读取两个字节并将它们重新组合在一起

unsigned adcval = RX() << 8;
adcval |= RX();