OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 6833|回复: 16

重写串口printf()函数代码分享

[复制链接]

3

主题

312

帖子

0

精华

高级会员

Rank: 4

积分
907
金钱
907
注册时间
2011-10-19
在线时间
196 小时
发表于 2018-9-1 13:49:11 | 显示全部楼层 |阅读模式
本帖最后由 ufbycd 于 2018-9-1 14:10 编辑

看到有人写了分享贴,吓我一跳,以为是分享printf实现!进去一看却不是!还是我来吧!
[mw_shl_code=c,true]

/*
BYTE_OUT is the only external dependency for this file,
if you have a working BYTE_OUT, leave it commented out.
If not, uncomment the define below and
replace outbyte(c) by your own function call.

#define BYTE_OUT(c) outbyte(c)
*/

#include "serial.h"
#include <stdarg.h>

#define BYTE_OUT(c)  Serial_PutChar(c)

#define SUPPOR_LONG_LONG          0
#define PRINT_FLOAT_AGINST_DOUBLE 0
#if ! PRINT_FLOAT_AGINST_DOUBLE
#        undef SUPPOR_LONG_LONG
#        define SUPPOR_LONG_LONG 1
#endif

struct _mbuf
{
    int isSizeLimited;
    size_t size;
    char *data;
};

static int printchar(struct _mbuf *sbuf, int c)
{
    if (sbuf)
    {
        if (sbuf->isSizeLimited)
        {
            if (sbuf->size < 1)
                return -1;
            else if (sbuf->size == 1)
            {
                *(sbuf->data) = '\0';
                sbuf->size = 0;
                return 1;
            }
        }

        *(sbuf->data) = (char) c;
        sbuf->data += 1;
        sbuf->size -= 1;
    }
    else
    {
        BYTE_OUT((uint8_t)c);
    }

    return 1;
}

#define PAD_RIGHT 1
#define PAD_ZERO 2

static int prints(struct _mbuf *sbuf, const char *string, int width, int pad)
{
    register int pc = 0, padchar = ' ';

    if (width > 0)
    {
        register int len = 0;
        register const char *ptr;
        for (ptr = string; *ptr; ++ptr)
            ++len;
        if (len >= width)
            width = 0;
        else
            width -= len;
        if (pad & PAD_ZERO)
            padchar = '0';
    }
    if (!(pad & PAD_RIGHT))
    {
        for (; width > 0; --width)
        {
            printchar(sbuf, padchar);
            ++pc;
        }
    }
    for (; *string; ++string)
    {
        printchar(sbuf, *string);
        ++pc;
    }
    for (; width > 0; --width)
    {
        printchar(sbuf, padchar);
        ++pc;
    }

    return pc;
}

/* the following should be enough for 32 bit int */
#define PRINT_BUF_LEN 24

static int printInt(struct _mbuf *sbuf, int value, int ary, int sg, int width,
        int pad, int letbase)
{
    char print_buf[PRINT_BUF_LEN];
    register char *str;
    register int t, neg = 0, pc = 0;
    register unsigned int u = value;

    if (value == 0)
    {
        print_buf[0] = '0';
        print_buf[1] = '\0';
        return prints(sbuf, print_buf, width, pad);
    }

    if (sg && (ary == 10) && (value < 0))
    {
        neg = 1;
        u = -value;
    }

    str = print_buf + PRINT_BUF_LEN - 1;
    *str = '\0';

    while (u)
    {
        t = u % ary;
        if (t >= 10)
            t += letbase - '0' - 10;
        *--str = t + '0';
        u /= ary;
    }

    if (neg)
    {
        if (width && (pad & PAD_ZERO))
        {
            printchar(sbuf, '-');
            ++pc;
            --width;
        }
        else
        {
            *--str = '-';
        }
    }

    return pc + prints(sbuf, str, width, pad);
}

#if SUPPOR_LONG_LONG
static int printLongLongInt(struct _mbuf *sbuf, long long int value, int ary,
        int sg, int width, int pad, int letbase)
{
    char print_buf[PRINT_BUF_LEN];
    register char *str;
    register int t, neg = 0, pc = 0;
    register unsigned long long int u = value;

    if (value == 0)
    {
        print_buf[0] = '0';
        print_buf[1] = '\0';
        return prints(sbuf, print_buf, width, pad);
    }

    if (sg && ary == 10 && value < 0)
    {
        neg = 1;
        u = -value;
    }

    str = print_buf + PRINT_BUF_LEN - 1;
    *str = '\0';

    while (u)
    {
        t = u % ary;
        if (t >= 10)
            t += letbase - '0' - 10;
        *--str = t + '0';
        u /= ary;
    }

    if (neg)
    {
        if (width && (pad & PAD_ZERO))
        {
            printchar(sbuf, '-');
            ++pc;
            --width;
        }
        else
        {
            *--str = '-';
        }
    }

    return pc + prints(sbuf, str, width, pad);
}
#else
#        define printLongLongInt(sbuf, value, ary, sg, width, pad, letbase) 0
#endif

#if ! PRINT_FLOAT_AGINST_DOUBLE
static int printDouble(struct _mbuf *sbuf, double value, int ary, int width,
        int precision, int pad, int letbase)
{
    int num;
    long long int llvalue, lli;
    unsigned long long ull;
    int i;

    num = 0;

    llvalue = (long long int) value;
    num += printLongLongInt(sbuf, llvalue, ary, 1, width, pad, letbase);

    printchar(sbuf, '.');
    num += 1;

    lli = 1;
    for (i = 0; i < precision; i++)
    {
        lli *= 10;
    }

    value -= (double)llvalue;
    llvalue = (long long int) (value * lli); // 加 0.5 作四舍五入
    ull = (llvalue > 0) ? llvalue : -llvalue;
    num += printLongLongInt(sbuf, ull, ary, 1, precision, PAD_ZERO, letbase);

    return num;
}
#else
static int printFloat(struct _mbuf *sbuf, float value, int ary, int width, int precision, int pad, int letbase)
{
    int num;
    int32_t llvalue, lli;
    uint32_t uli;
    int i;

    num = 0;

    llvalue = (int32_t) value;
    num += printInt(sbuf, llvalue, ary, 1, width, pad, letbase);

    printchar(sbuf, '.');
    num += 1;

    lli = 1;
    for(i = 0; i < precision; i++)
    {
        lli *= 10;
    }

    value -= llvalue;
    llvalue = (int32_t) (value * lli);
    uli = (llvalue > 0) ? llvalue : - llvalue;
    num += printInt(sbuf, uli, ary, 1, precision, PAD_ZERO, letbase);

    return num;
}
#endif

/**
* @brief A format specifier follows this prototype:
* %[flags][width][.precision][length]specifier
*
* @param out
* @param format
* @param args
* @return
*/
static int print(struct _mbuf *sbuf, const char *format, va_list args)
{
    register int width, precision, pad;
    register int num = 0;
    char scr[2];
    char c, *pchar;

    enum _length
    {
        _normal = 0, _short, _short_short, _long, _long_long
    } length;

    for (; *format != 0; ++format)
    {

        if (*format != '%')
        {
            printchar(sbuf, *format);
            ++num;
        }
        else if (format[1] == '%')
        {
            printchar(sbuf, '%');
            ++format;
            ++num;
        }
        else
        {
            ++format;
            width = 0;
            precision = 0;
            length = _normal;
            pad = 0;

            c = *format;
            if (c == '\0')
                break;

            /* parse flags */
            switch (c)
            {
            case '-':
                pad = PAD_RIGHT;
                ++format;
                break;
            case '+':
            case '#':
                ++format;
                break;
            case '0':
                pad |= PAD_ZERO;
                ++format;
                break;
            }

            /* Parse width */
            while ((*format >= '0') && (*format <= '9'))
            {
                width *= 10;
                width += *format - '0';
                ++format;
            }

            /* Parse precision */
            if (*format == '.')
            {
                ++format;
                while ((*format >= '0') && (*format <= '9'))
                {
                    precision *= 10;
                    precision += *format - '0';
                    ++format;
                }
            }
            else
            {
                precision = 6;
            }

            /* parse length */
            switch (*format)
            {
            case 'l':
                if (format[1] == 'l')
                {
                    length = _long_long;
                    ++format;
                }
                else
                    length = _long;

                ++format;
                break;

            case 'h':
                if (format[1] == 'h')
                    format += 2;
                break;

            case 'L':
            case 'j':
            case 'z':
            case 't':
                ++format;
                break;
            }

            int letbase;
            if (*format >= 'a')
                letbase = 'a';
            else
                letbase = 'A';

            /* parse specifier */
            switch (*format)
            {
            case 's':
                pchar = (char *) va_arg(args, int);
                num += prints(sbuf, pchar ? pchar : "(null)", width, pad);
                break;

            case 'd':
                if (length == _long_long)
                    num += printLongLongInt(sbuf, va_arg(args, long long int),
                            10, 1, width, pad, letbase);
                else if (length == _long)
                    num += printInt(sbuf, va_arg(args, long int), 10, 1, width,
                            pad, letbase);
                else
                    num += printInt(sbuf, va_arg(args, int), 10, 1, width, pad,
                            letbase);
                break;

            case 'x':
            case 'X':
                if (length == _long_long)
                    num += printLongLongInt(sbuf,
                            va_arg(args, long long int), 16, 0, width,
                            pad, letbase);
                else if (length == _long)
                    num += printInt(sbuf, va_arg(args, long int), 16,
                            0, width, pad, letbase);
                else
                    num += printInt(sbuf, va_arg(args, int), 16, 0,
                            width, pad, letbase);
                break;

            case 'u':
            case 'p':
                if (length == _long_long)
                    num += printLongLongInt(sbuf,
                            va_arg(args, long long int), 10, 0, width,
                            pad, letbase);
                else if (length == _long)
                    num += printInt(sbuf, va_arg(args, long int), 10,
                            0, width, pad, letbase);
                else
                    num += printInt(sbuf, va_arg(args, int), 10, 0,
                            width, pad, letbase);
                break;

            case 'o':
                if (length == _long_long)
                    num += printLongLongInt(sbuf,
                            va_arg(args, long long int), 8, 0, width,
                            pad, letbase);
                else if (length == _long)
                    num += printInt(sbuf, va_arg(args, long int), 8, 0,
                            width, pad, letbase);
                else
                    num += printInt(sbuf, va_arg(args, int), 8, 0,
                            width, pad, letbase);
                break;

            case 'c':
                /** @note char are converted to int then pushed on the stack */
                scr[0] = (char) va_arg(args, int);
                scr[1] = '\0';
                num += prints(sbuf, scr, width, pad);
                break;

            case 'f':
            case 'F':
            case 'e':
            case 'E':
            case 'g':
            case 'G':
#if PRINT_FLOAT_AGINST_DOUBLE
                num += printFloat(sbuf, va_arg(args, float), 10, width, precision, pad, letbase);
#else
                num += printDouble(sbuf, va_arg(args, double), 10, width,
                        precision, pad, letbase);
#endif
                break;
            }
        }
    }

    if (sbuf)
        printchar(sbuf, '\0');

    va_end(args);

    return num;
}

int printf (const char *format, ...)
{
    int ret;
    va_list args;

    va_start(args, format);
    ret = print(NULL, format, args);
    va_end(args);

    return ret;
}

int vprintf(const char *format, va_list args)
{
    return print(NULL, format, args);
}

int sprintf(char *out, const char *format, ...)
{
    va_list args;
    struct _mbuf mybuf =
    { 0, (size_t) -1, out };

    va_start(args, format);
    return print(&mybuf, format, args);
}

int vsprintf(char *out, const char *format, va_list args)
{
    struct _mbuf mybuf =
    { 0, (size_t) -1, out };

    return print(&mybuf, format, args);
}

int snprintf(char *out, unsigned int size, const char *format, ...)
{
    va_list args;
    struct _mbuf mybuf =
    { 1, size, out };

    va_start(args, format);
    return print(&mybuf, format, args);
}

int puts(const char *s)
{
    int i;
    char c;

    c = *s++;
    for (i = 0; c != '\0'; i++)
    {
        BYTE_OUT((uint8_t)c);
        c = *s++;
    }

    BYTE_OUT('\n');

    return i + 1;
}

[/mw_shl_code]
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

109

主题

5564

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
10570
金钱
10570
注册时间
2017-2-18
在线时间
1913 小时
发表于 2018-9-1 14:33:35 | 显示全部楼层
回复 支持 反对

使用道具 举报

3

主题

312

帖子

0

精华

高级会员

Rank: 4

积分
907
金钱
907
注册时间
2011-10-19
在线时间
196 小时
 楼主| 发表于 2018-9-1 22:17:48 | 显示全部楼层
自己顶一下。
因为感觉C库的printf体积比自己实现的要大挺多,所以实际工程项目都是用这个printf实现。
回复 支持 反对

使用道具 举报

9

主题

165

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
251
金钱
251
注册时间
2018-4-6
在线时间
31 小时
发表于 2018-9-2 10:52:18 | 显示全部楼层
顶一下
回复 支持 反对

使用道具 举报

31

主题

130

帖子

0

精华

高级会员

Rank: 4

积分
790
金钱
790
注册时间
2015-6-28
在线时间
139 小时
发表于 2018-9-10 11:10:32 | 显示全部楼层
把楼主上面的代码修改串口打印添加进去之后,将原来的printf重定向屏蔽掉,然后代码就运行不起来了,请问这个怎么解决?
回复 支持 反对

使用道具 举报

3

主题

312

帖子

0

精华

高级会员

Rank: 4

积分
907
金钱
907
注册时间
2011-10-19
在线时间
196 小时
 楼主| 发表于 2018-9-10 20:53:05 | 显示全部楼层
黑猫 发表于 2018-9-10 11:10
把楼主上面的代码修改串口打印添加进去之后,将原来的printf重定向屏蔽掉,然后代码就运行不起来了,请问这 ...

不用屏蔽原来的重定向,如果不屏蔽就有输出,说明还是链接到C库里的printf了
回复 支持 反对

使用道具 举报

3

主题

102

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
228
金钱
228
注册时间
2018-9-7
在线时间
22 小时
发表于 2018-9-11 09:44:13 | 显示全部楼层
谢谢分享,学习了
像屎壳螂一样做人,在粪斗中成长......
回复 支持 反对

使用道具 举报

1

主题

33

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
207
金钱
207
注册时间
2018-4-29
在线时间
48 小时
发表于 2018-9-11 09:49:43 | 显示全部楼层
谢谢大佬
回复 支持 反对

使用道具 举报

31

主题

130

帖子

0

精华

高级会员

Rank: 4

积分
790
金钱
790
注册时间
2015-6-28
在线时间
139 小时
发表于 2018-9-11 10:14:36 | 显示全部楼层
ufbycd 发表于 2018-9-10 20:53
不用屏蔽原来的重定向,如果不屏蔽就有输出,说明还是链接到C库里的printf了

如果是不屏蔽,使用printf还是用的是原来的C库,仿真压根没调用到自己写的呢。不懂怎么搞
回复 支持 反对

使用道具 举报

3

主题

312

帖子

0

精华

高级会员

Rank: 4

积分
907
金钱
907
注册时间
2011-10-19
在线时间
196 小时
 楼主| 发表于 2018-9-11 17:03:54 | 显示全部楼层
黑猫 发表于 2018-9-11 10:14
如果是不屏蔽,使用printf还是用的是原来的C库,仿真压根没调用到自己写的呢。不懂怎么搞

去掉链接器里的printf相关的选项。
实在不行的话,就把自定义的printf改名吧,再用一个宏指向这个函数:
如将自定义的"printf”改名为mprintf,再定义这个宏: #define printf mprintf
来使用

原理是宏名的优先级高于函数名
回复 支持 反对

使用道具 举报

35

主题

73

帖子

0

精华

初级会员

Rank: 2

积分
88
金钱
88
注册时间
2018-4-10
在线时间
49 小时
发表于 2018-9-11 22:51:20 | 显示全部楼层
插一个眼,以后再看
回复 支持 反对

使用道具 举报

31

主题

130

帖子

0

精华

高级会员

Rank: 4

积分
790
金钱
790
注册时间
2015-6-28
在线时间
139 小时
发表于 2018-9-12 20:23:18 | 显示全部楼层
ufbycd 发表于 2018-9-11 17:03
去掉链接器里的printf相关的选项。
实在不行的话,就把自定义的printf改名吧,再用一个宏指向这个函数: ...

去掉链接器里的printf相关的选项。 这句不知道怎么弄
回复 支持 反对

使用道具 举报

6

主题

315

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1669
金钱
1669
注册时间
2018-1-29
在线时间
160 小时
发表于 2018-9-13 08:56:21 | 显示全部楼层
感谢分享, printf占用空间太大了!
回复 支持 反对

使用道具 举报

15

主题

96

帖子

0

精华

高级会员

Rank: 4

积分
650
金钱
650
注册时间
2017-4-21
在线时间
151 小时
发表于 2018-11-9 10:30:56 | 显示全部楼层
这个执行一句printf函数会耗时多少?
回复 支持 反对

使用道具 举报

0

主题

2

帖子

0

精华

新手入门

积分
10
金钱
10
注册时间
2019-5-9
在线时间
1 小时
发表于 2019-5-10 19:42:09 | 显示全部楼层
找不到serial.h咋办
回复 支持 反对

使用道具 举报

2

主题

37

帖子

0

精华

初级会员

Rank: 2

积分
114
金钱
114
注册时间
2012-2-29
在线时间
14 小时
发表于 2019-9-4 15:41:49 | 显示全部楼层
太牛了!谢谢!
回复 支持 反对

使用道具 举报

5

主题

66

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
232
金钱
232
注册时间
2016-10-20
在线时间
42 小时
发表于 2019-11-4 22:06:42 | 显示全部楼层
66666
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-5-29 04:40

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表