如何将__DATE__和__TIME__宏拆分为各个组件以进行变量声明?

时间:2017-10-23 22:00:27

标签: c++ c date macros c-preprocessor

我有以下结构(在嵌入式系统上):

struct Calib_Time_struct
{
    uint16_t    year;
    uint16_t    month;
    uint16_t    day;
    uint16_t    hour;
    uint16_t    minute;
    uint16_t    seconds;
};

我有一个"默认" values数组,我需要在其中插入时间值:

struct Calib_Table_struct
{
    unsigned int                    table_id;           //!< Table identifier.
    char                            group_name[CAL_TBL_MAX_GROUP_NAME_LENGTH];
    unsigned int                    channel_number;     //!< Channel number within the group.
    float                           floor_value;        //!< Minimum value for a channel.
    unsigned int                    size;               //!< Number of elements in the table.
    struct Calib_Time_struct    modification_date;  //!< Date of modification.
};
static const struct Calib_Time_struct default_values[] =
{
    // Table 0
    {
        .table_id = 0U,
        .group_name = "ADC",
        .channel_number = 0U,
        .floor_value = 0.0f,
        .size = 1,
        .modification_date =
        {
            .year       = /* extract from __DATE__ macro */;
        },
    },
    //...
};

我想填写&#34; modification_date&#34;的年份,月份和日期。来自__DATE__宏的成员。

有没有办法做到这一点? (任何黑客?)

是否可以将类似方法或黑客应用于__TIME__宏?

动机是允许编译器(在构建服务器上)自动插入值,而不是让开发人员这样做。我们团队中有很多开发人员,并使用构建服务器来制作&#34;官方&#34;向我们团队以外的人提供的构建。

数据将附加到可执行文件并存储在Flash中,由引导加载程序下载到内存中。

默认数组中有很多(超过80个)表。

我的工具:
IAR Systems IDE&amp;编译器:7.4
使用ARM Cortex-A8的嵌入式系统平台。

语言:主要在C语言中,但对C ++语言人员可能有用。

2 个答案:

答案 0 :(得分:3)

C规范保证__DATE__将是

形式的字符串文字
MMM DD YYYY

如果日期是单个字符长,则日期的第一个数字为空格。这意味着你可以做这样的残忍和不寻常的事情,从技术上说这是不可移植的(预处理器不必使用与目标相同的字符编码系统),但应该可以工作:

#define YEAR_CHAR 7
#define YEAR  (\
                ((__DATE__)[YEAR_CHAR + 0] - '0') * 1000 + \
                ((__DATE__)[YEAR_CHAR + 1] - '0') * 100  + \
                ((__DATE__)[YEAR_CHAR + 2] - '0') * 10   + \
                ((__DATE__)[YEAR_CHAR + 3] - '0') * 1      \
              )

__TIME__宏保证格式为

hh:mm:ss

所以你可以像这样做一些可怕的事情:

#define HOUR_CHAR 0
#define HOUR (\
                ((__TIME__)[HOUR_CHAR + 0] - '0') * 10 + \
                ((__TIME__)[HOUR_CHAR + 1] - '0') * 1    \
             )

获取小时,分钟或秒。

如果你有一个符合C ++ 14的编译器,你可以使用constexpr函数来计算这个不那么可怕的:

constexpr int compilationYear() {
    const int kYearChar = 7;
    const int kNumYearChars = 4;

    int result = 0;
    for (int i = yearChar + kNumYearChars - 1, multiplier = 1;
         i >= kYearChar;
         i--, multiplier *= 10) {
        result += (__DATE__[i] - '0') * multiplier;
    }

    return result;
}

然后你可以赋值compilationYear(),并在编译时评估该函数。

答案 1 :(得分:1)

您可以使用对__DATE__宏的解析函数的运行时调用来简单地初始化它(»Mmm dd yyyy«,这意味着年份的字符位于已知位置):

int extractYearFromDateMacro() {
    return 1000 * (__DATE__[7] - '0') + 100 * (__DATE__[8] - '0') + 10 * (__DATE__[9] - '0') + (__DATE__[10] - '0');
}
    .modification_date =
    {
        .year       = extractYearFromDateMacro()
    },

从C ++ 11开始,您可以在前面添加constexpr并让编译器对其进行评估。 (也用于常量表达式)

Demo

同样的方法也可以应用于__TIME__宏(“hh:mm:ss”)。

如果只连接表达式,使用预处理器甚至不能保证编译时评估。这些仍可在运行时进行评估。

上面的

extractYearFromDateMacro可能会以与PP扩展结果相同的方式不断折叠。

相反,使用constexpr方法将保证编译时评估。