可变参数函数,原理,实现,UART寄存器操作

今天再次遇到可变参数的函数的实现,突然发现自己还不是那么清楚。唉,治学不严谨啊。所以这次一定要彻底弄明白。

这篇文章已经说的非常清楚了CSDN C/C++电子杂志第一期 之 可变参数学习笔记 ,所以我就拿今天看到的代码来说说吧

/*****************************************************************************
*
*
*   @func   void    |   FormatString | Simple formatted output string routine
*
*   @rdesc  Returns length of formatted string
*
*   @parm   unsigned char * |   pBuf |
*               Pointer to string to return formatted output.  User must ensure
*               that buffer is large enough.
*
*   @parm   const unsigned char * |   sz,... |
*               Format String:
*
*               @flag Format string | type
*               @flag u | unsigned
*               @flag d | int
*               @flag c | char
*               @flag s | string
*               @flag x | 4-bit hex number
*               @flag B | 8-bit hex number
*               @flag H | 16-bit hex number
*               @flag X | 32-bit hex number
*
*   @comm
*           Same as OutputFormatString, but output to buffer instead of serial port.
*/
unsigned int FormatString (unsigned char *pBuf, const unsigned char *sz, ...)
{
    unsigned char    c;
    va_list         vl;      //申明一个va_list(char *或者void *)的变量
    
    va_start(vl, sz);   //得到最后一个固定参数的地址
   
    szSprintf = pBuf;//char *szSprintf;一个全局的char *
    while (*sz) {      
        c = *sz++;
        switch (c) {
        case '%':
            c = *sz++;
            switch (c) {  //switch(*++sz)
            case 'x':     //4-bit hex number
                pOutputNumHex(va_arg(vl, unsigned long), 0);
                break;
            case 'B':     //8-bit hex number
                pOutputNumHex(va_arg(vl, unsigned long), 2);
                break;
            case 'H':    //16-bit hex number
                pOutputNumHex(va_arg(vl, unsigned long), 4);
                break;
            case 'X':      //32-bit hex number
                pOutputNumHex(va_arg(vl, unsigned long), 8);
                break;
            case 'd':    //int
                {
                    long    l;
               
                    l = va_arg(vl, long);
                    if (l < 0) {
                        pOutputByte('-');
                        l = - l;
                    }
                    pOutputNumDecimal((unsigned long)l);
                }
                break;
            case 'u': //unsigned
                pOutputNumDecimal(va_arg(vl, unsigned long));
                break;
            case 's':  //string
                OutputString(va_arg(vl, char *));
                break;
            case '%': //直接打印出%
                pOutputByte('%');
                break;
            case 'c': //char
                c = va_arg(vl, unsigned char);
                pOutputByte(c);
                break;
               
            default:
                pOutputByte(' ');
                break;
            }
            break;
        case '/r':
            if (*sz == '/n')
                sz ++;
            c = '/n';
            // fall through
        case '/n':
            pOutputByte('/r');
            // fall through
        default:
            pOutputByte(c);
        }
    }
    pOutputByte(0);
    c = szSprintf - pBuf;
    szSprintf = 0;
    va_end(vl);   //良好习惯,vl=NULL,以防止以后的误操作
    return (c);
}

/*****************************************************************************
*
*
*   @func   void    |   pOutputByte | Sends a byte out of the monitor port.
*
*   @rdesc  none
*
*   @parm   unsigned int |   c |
*               Byte to send.
*
*/

static void
pOutputByte(
    unsigned char c
)

{
    if (szSprintf)
        *szSprintf++ = c;
    else
        OEMWriteDebugByte(c);
}


/*****************************************************************************
*
*
*   @func   void    |   pOutputNumHex | Print the hex representation of a number through the monitor port.
*
*   @rdesc  none
*
*   @parm   unsigned long |   n |
*               The number to print.
*
*   @parm   long | depth |
*               Minimum number of digits to print.
*
*/

static void pOutputNumHex (unsigned long n, long depth)
{
    if (depth) {
        depth--;
    }
   
    if ((n & ~0xf) || depth) {
        pOutputNumHex(n >> 4, depth);
        n &= 0xf;
    }
   
    if (n < 10) {
        pOutputByte((unsigned char)(n + '0'));
    } else {
    pOutputByte((unsigned char)(n - 10 + 'A'));
}
}


/*****************************************************************************
*
*
*   @func   void    |   pOutputNumDecimal | Print the decimal representation of a number through the monitor port.
*
*   @rdesc  none
*
*   @parm   unsigned long |   n |
*               The number to print.
*
*/

static void pOutputNumDecimal (unsigned long n)
{
    if (n >= 10) {
        pOutputNumDecimal(n / 10);
        n %= 10;
    }
    pOutputByte((unsigned char)(n + '0'));
}

/*****************************************************************************
*
*
*   @func   void    |   OutputString | Sends an unformatted string to the monitor port.
*
*   @rdesc  none
*
*   @parm   const unsigned char * |   s |
*               points to the string to be printed.
*
*   @comm
*           backslash n is converted to backslash r backslash n
*/

static void OutputString (const unsigned char *s)
{
    while (*s) {       
        if (*s == '/n') {
            OEMWriteDebugByte('/r');
        }
        OEMWriteDebugByte(*s++);
    }
}


/*****************************************************************************
*
*
*   @func   void    |   OEMWriteDebugString | Display string to the monitor port.
*
*   @parm   unsigned short * | str |
*           Points to the receiving buffer.
*/

void
OEMWriteDebugString(unsigned short *str)

{
 // Loop through text string, sending characters.
 //

    while (str && *str)
 {
        OEMWriteDebugByte((unsigned char)*str++);
 }
}

/*****************************************************************************
*
*
*   @func   void    |   OEMWriteDebugByte | Output byte to the monitor port.
*
*   @parm   unsigned char *| str |
*           Points to the output buffer.
*/

void
OEMWriteDebugByte(UCHAR ch)

{
    volatile UART0reg *s2410UART0 = (UART0reg *)UART0_BASE;   //对寄存器设置

 // Wait for transmit buffer to be empty.
 //

 while(!(s2410UART0->rUTRSTAT & 0x2))
 {
 }

    s2410UART0->rUTXH = ch;    //对寄存器“赋值”
}


int
OEMReadDebugByte()

{
    unsigned char ch;
    volatile UART0reg *s2410UART0 = (UART0reg *)UART0_BASE;
   
 // Any receive data for us?
 //

    if (!(s2410UART0->rUTRSTAT & 0x1))
 {
        ch = OEM_DEBUG_READ_NODATA;  // No data.
 }
    else
 {
        ch = s2410UART0->rURXH;   // Read character.
 }

    return (ch);
}

关键字:可变参数函数,原理,实现,UART寄存器操作

//;UART0收发一组字符(48个)(通过) // rxd_str必须是全局变量,切使用前要赋值。 // rxd_str不能是局部变量,局部变量会清零, //因为中断一次只能收一个字符rxd_str不可能>=LENM /********************************************** 文件描述: 三相电测量上报 功能说明: 测量ATT7022B完成 PCF8563上报定时(需校时) FM24C256转换数据存储 UART0口用于RS232读数据 UART1控制无线模块上报数据 创建:2006年12月5日 /********************************* *函数:main. *入口:无 *出口:无 *功能说明:接收UART端口命令 执行读ATT7022B的数据 进行处理、存储和回发数据 ********************************/ //***************************** //包含的文件 //****************************** #include <c8051f020.h> #include <main.h> #include <intrins.h> //************************************************* //函数:init_sysclk(void) //功能:时钟初始化 //入口:无 //出口: 无 //说明:使用外部时钟12M //************************************************* void init_sysclk (void) { uint i=0; OSCXCN=0x67; //external oscillator with 12MHz crystal for(i=0;i<256;i++); // XTLVLD blanking while(!(OSCXCN & 0x80)); // Wait for crystal osc. to settle OSCICN=0x88; //时钟丢失检测,选择外部时钟 CKCON = 0x00; //时钟分频 } /********************************* 函数:void enable_wdog(void) void disable_wdog(void) 功能:使能、禁止wdog 入口:无 出口:无 说明: ********************************/ void enable_wdog(void) { WDTCN=0xA5; //允许看门狗定时器工作 } void disable_wdog(void) //禁止看门狗定时器工作 { WDTCN=0xDE; WDTCN=0xAD; } /*********************************** 函数:void init_ioport() 功能:端口配置及端口位定义 入口:无 出口:无 说明: *************************************/ void init_ioport() { XBR0=0x07; //TXD0-P0.0 RXD0-P0.1,SPI_SCK-P0.2,SPI_MISO-P0.3 //SPI_MOSI-P0.4,SPI_NSS-P0.5,SDA-P0.6,SCL-P0.7, XBR2=0x44; //端口I/O弱上拉允许,TX1-P1.0,RXD-P1.1交叉开关允许 XBR1=0X10; //INT1使能INT1--P1.2 P0MDOUT=0x00; //端口0输出方式寄存器:0--漏极开路 P0=0xff; P1MDIN=0XFF; //端口1输入方式寄存器:0--配置为数字输入 P1MDOUT=0x00; //端口1输出方式寄存器,0--漏极开路 P1=0xff; P2MDOUT=0x00; //端口2输出方式寄存器:0--漏极开路 P2=0xff; P3MDOUT=0X00; //端口3输出方式寄存器:0--漏极开路 P3=0xff; P74OUT=0x00; //端口7-4输出方式寄存器:0--漏极开路 P4=0xff; P5=0xff; P6=0xff; P7=0xff; } //-------------------------------------------------------------- //函数:void Delay_ms (unsigned ms) //功能:实现延时功能 Timer0_ms //--------------------------------------------------------------- // /* Configure Timer0 to delay <ms> */ void Delay_ms (unsigned ms) { uchar i; // millisecond counter TCON &= ~0x30; // STOP Timer0 and clear overflow flag TMOD |= 0x01; // configure Timer0 to 16-bit mode CKCON |= 0x08; // Timer0 counts SYSCLKs for (i = 0; i < ms; i++) // count milliseconds { TR0 = 0; // STOP Timer0 TH0 = (-SYSCLK/1000) >> 8; // set Timer0 to overflow in 1ms TL0 = -SYSCLK/1000; TR0 = 1; // START Timer0 while (TF0 == 0); // wait for overflow TF0 = 0; // clear overflow indicator } } //------------------------------------------------------- //函数:void Delay_us (unsigned us) //功能:实现延时功能 Timer0_us //------------------------------------------------------- // /* Configure Timer0 to delay <us>*/ void Delay_us (unsigned us) { uchar i; // microseconds counter TCON &= ~0x30; // STOP Timer0 and clear overflow flag TMOD |= 0x01; // configure Timer0 to 16-bit mode CKCON |= 0x08; // Timer0 counts SYSCLKs for (i = 0; i < us; i++) // count microseconds { TR0 = 0; // STOP Timer0 TH0 = (-SYSCLK/1000000) >> 8; // set Timer0 to overflow in 1us TL0 = -SYSCLK/1000000; TR0 = 1; // START Timer0 while (TF0 == 0); // wait for overflow TF0 = 0; // clear overflow indicator } } //*************************************** //函数:unsigned char my_add(uchar my_add) //功能:读开关状态确定子地址 //入口:无 //出口:子地址 //说明:子地址存于myadd中(即设备号) //**************************************** void my_add(void) { P74OUT |=0XD0; myadd =P5; } //****************************************** //函数:void jiaob(ATT_JB[]) //功能:写校表寄存器 //说明:UART口接收校表时间并存于FM24C256中 //***************************************** void jiaob (void) { } main() { disable_wdog(); //关看门狗 init_sysclk (); //时钟初始化 init_ioport(); //交叉开关配置 my_add(); //读设备子地址设置 UART0_Init(); //UART0初始化 EX1= 1; //开INT1 EA = 1; //开中断 rxd_str=0; while(1) { if(uart0_flag) { rxd_str=0; uart0_flag = 0; m=ur0_rxd; txd_string(m,LENM); } } //---------判断本设备命令及命令内容执行命令------- } //----------------------------------------------------- //串口初始化 //-------------------------------------------------------- void UART0_Init(void) { SCON0 = 0x50; //串口方式1,波特率可变 PCON |= 0x00; //SMOD = 0 TMOD = 0x20; //选择T1方式2, TH1 = 0xe8; //T1初值, TL1 = 0xe8; ES0 = 1; //UART0中断开启 TR1 = 1; //启动定时器T1 } //---------------------------------------------------------- //发送单个字符 //--------------------------------------------------------- void txd_char(unsigned char ch) { SBUF0 = ch; //送入缓冲区 while(TI0 == 0); //等待发送完毕 TI0 = 0; //软件清零 } //----------------------------------------------------- //发送字符串,调用Send_Char() len字符串长度 //---------------------------------------------------- void txd_string(unsigned char * str,unsigned char len) { unsigned char k = 0; do { txd_char(*(str+k) ); k++; } while(k < len); } //-------------------------------------------------------- //UART0中断服务程序. 接收字符 //-------------------------------------------------------- // rxd_str必须是全局变量,切使用前要赋值。 // rxd_str不能是局部变量,局部变量会清零, //因为中断一次只能收一个字符rxd_str不可能>=LENM void uart0_isr(void) interrupt 4 using 1 { unsigned char rxch; if(RI0) //中断标志 RI0=1 数据完整接收 { RI0 = 0; //软件清零 rxch = SBUF0; //读缓冲 if(rxd_str>=LENM) { uart0_flag=1; rxd_str=0; } ur0_rxd[rxd_str] = rxch; //存入数组,供发送 rxd_str++; } } //********************** //main.h //********************* #ifndef _main_h #define _main_h //***************************** //全局常量 //**************************** #define uchar unsigned char #define uint unsigned int #define SYSCLK 11059200 #define CMD_RESET 0X11 #define CMD_TIME 0X12 #define CMD_DATA 0X13 #define ATT_R 0x00 // ATT Read command #define ATT_W 0x80 // ATT Write command #define fm_Write_add 0xA0 #define fm_Read_add 0xA1 #define Fm_add 0xA0 //fm24c256器件从地?#define LENM 0x30 unsigned char idata r_commond=0x01; //读命令代码 unsigned char idata w_time_commond=0x02; //校时代码 unsigned char xdata jb_commond=0xdd; //校表命令代码 //---------------------------------------------------- //全局变量 //---------------------------------------------------- unsigned char rxd_str; unsigned char * m; unsigned char xdata fm_read_buf[60]={0}; unsigned char xdata fm_write_buf[60]={0}; unsigned char xdata pcf_d[16]={0}; //data of pcf8563 unsigned char xdata *d_ptr; unsigned char xdata ur0_rxd[60]; //每次接收字符串 unsigned char xdata ur0_txd[60]; //要发送的字符串 unsigned char xdata att_rd[90]; unsigned char xdata att_wd[60]; // att7022 of data unsigned char xdata att_jb[40]; //校表数组(myadd=0) unsigned char idata period; //时间间隔 unsigned char idata myadd; unsigned char *str; unsigned char chksum; unsigned char ATT_W_ADD; // ATT Read status register unsigned char ATT_R_ADD; // ATT Write status register uchar slave_add,fm_ram_add,send_byte,write_num,read_num; uchar fm_send_count,fm_receive_count,fm_send_len,fm_receive_len; sbit SDA=P1^6; //*模拟I2C数据传送位*/ sbit SCL=P1^7; //*模拟I2C时钟控制位*/ sbit ATT_CS = P3^0; // ATT CS signal sbit D_E = P1^3; //485收发控制?bit ack; bit uart0_flag; //中断接收完成标志 bit uart1_flag; //中断接收完成标志 bit sm_busy; //收发开始置1,操作结束后由中断清0 bit fm_err_flag; //--------子函数声明------------------------------------- void uart0_isr(); //串口中断服务程序,接收字符 void UART0_Init(void); void txd_char(uchar ch); void txd_string(uchar * str,uchar len); void init_sysclk (void); void enable_wdog(void); void disable_wdog(void); void init_ioport(); void Delay_ms (unsigned ms); void Delay_us (unsigned us) ; void my_add(void); void jiaob (void); void ATT_Write (uchar ATT_WADD, uchar att_wd[]); unsigned char ATT_Read (uchar ATT_RADD,uchar att_rd[]); void SPI0_Init (void); void Start_I2c(); void Stop_I2c(); void Ack_I2c(); void SendByte(uchar x); uchar RcvByte(); bit ISendStr(uchar sla,uchar suba,uchar *d_ptr,uchar no); bit IRcvStr(uchar sla,uchar suba,uchar *d_ptr,uchar no); void WriteClock(void); void StartClock(void); void init_sysclk (void); void init_ioport(); void init_smbus(void); void smbus_receive (uchar chip_select,byte_address,receive_num); void smbus_send (uchar chip_select,byte_address,write_num); void enable_wdog(void); void disable_wdog(void); #endif
在单片机中,printf函数实现需要涉及串口通信和字符串处理。 下面是一个简单的示例代码,假设使用的是串口1进行通信,波特率为115200: ```c #include <stdio.h> // 定义串口1的地址 #define UART1_BASE 0x4000C000 // 定义UART寄存器结构体 typedef struct { unsigned int DR; // 数据寄存器 unsigned int RSR; // 接收状态寄存器 unsigned int FR; // 状态寄存器 unsigned int ILPR; // 红外线调制器预分频器寄存器 unsigned int IBRD; // 波特率整数部分寄存器 unsigned int FBRD; // 波特率小数部分寄存器 unsigned int LCR_H; // 控制寄存器高位 unsigned int CR; // 控制寄存器 unsigned int IFLS; // 中断触发寄存器 unsigned int IMSC; // 中断屏蔽寄存器 unsigned int RIS; // 中断状态寄存器 unsigned int MIS; // 中断屏蔽状态寄存器 unsigned int ICR; // 中断清除寄存器 } UART_TypeDef; // 定义UART1指针 UART_TypeDef* const UART1 = (UART_TypeDef*) UART1_BASE; // 定义printf函数 int printf(const char* format, ...) { char buffer[256]; va_list args; va_start(args, format); int len = vsnprintf(buffer, 256, format, args); va_end(args); for (int i = 0; i < len; i++) { while (UART1->FR & (1 << 5)); // 等待发送缓冲区为空 UART1->DR = buffer[i]; // 发送数据 } return len; } ``` 上述代码中,首先定义了一个UART_TypeDef结构体,用于表示UART寄存器集合。然后定义了一个UART1的指针,指向串口1的地址。接着定义了一个printf函数,该函数使用了可变参数列表,将格式化字符串输出到buffer缓冲区中。最后使用串口1将缓冲区中的数据发送出去。 需要注意的是,上述代码只是一个简单的示例,实际应用中需要根据具体的硬件平台和编译器来进行调整。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值