3

例如,有来源:

void func1() {
    func3();
    if(qqq) {
         func2();
    }
    func4(
    );
}

它应该转换为:

void func1() {
MYMACRO
    func3();
MYMACRO
    if(qqq) {
MYMACRO
         func2();
MYMACRO
    }
MYMACRO
    func4(
    );
MYMACRO
}

即在语句可以的每一行的末尾插入“MYMACRO\n”,仅在函数内部。

如何轻松做到?我应该使用正则表达式吗?我应该使用什么工具?

例如,gcc 可以在函数内部输出所有语句开始(或结束)的所有行号吗?

@related 如何告诉 gcc 通过调用我自己的函数来检测代码的每一行代码?

@related 我应该使用什么分析器来测量在这个函数中花费的_real_时间(包括等待系统调用),而不是_CPU_一个

4

3 回答 3

2

你想通过这样做来完成什么?根据任务的描述,可能有一种更简单的方法来解决问题。如果您确定这是完成任务的最佳方式,请继续阅读。


您必须实现某种基本的 C 语言解析器才能做到这一点。由于您正在处理文本,因此我建议您使用 perl、python 或 ruby​​ 等脚本语言来修改您的文本,而不是编写 C 程序来完成。

您的解析器将一次遍历文件一行,并且对于每一行,它将确定是否需要插入您的宏。解析器需要跟踪许多事情。首先,它需要跟踪它当前是否在评论中。当你遇到一个/*序列时,设置一个“in comment”标志并在下次遇到序列时清除它*/。无论何时设置该标志,您都不会添加宏调用。此外,您将需要跟踪您是否在函数内。假设您的代码相当简单明了,您可以有一个“大括号计数器”,它从零开始,遇到 a 时递增,遇到 a{时递减}. 如果您的大括号计数器为零,那么您不在函数内部,您不应该添加宏调用。您还需要添加特殊代码来检测和忽略作为结构定义、数组初始值设定项等的一部分的大括号。请注意,如果您的代码执行更复杂的操作,则简单的大括号计数将不起作用,例如:

void some_function (int arg) {
#ifdef CHECK_LIMIT_ONLY
    if (arg == 0) {
#else
    if (arg < 10) {
#endif
        // some code here
        ...
    }
}

虽然您可能会争辩说代码片段只是编写不佳的代码,但它只是您可能遇到的问题类型的一个示例。如果你的代码中有一些东西破坏了简单的大括号计数,那么这个问题就会变得更加困难。判断您的代码是否会中断大括号计数的一种方法是,您是否以非零大括号计数到达文件末尾,或者大括号计数是否在任何时间点变为负数。

一旦您可以确定您何时在函数中而不是在注释中,您需要确定该行是否需要在其后插入宏。您可以从一些简单的规则开始,测试脚本,看看是否有遗漏的情况。对于初学者,任何以分号结尾的行都是语句的结尾,您需要在它之后插入一个宏。与计算大括号类似,当您在函数内部时,您将需要计算括号,以便确定您是否在函数调用、循环条件或其他复合语句内部。如果您在其中之一内,则不会添加宏。要跟踪的另一个代码位置是{ ... }块的开始行和结束行。如果一行以{or结尾},您将在其后添加一个宏。

对于这样一个复杂的任务,你肯定会想编写一些脚本,在一段相对简单的代码上尝试一下,看看它出了什么问题。进行调整以涵盖您第一次错过的案例并重新测试。当它可以正确解析简单代码时,给它一些更复杂的东西,看看它的表现如何。

''更新:'' 为了解决一些人表达的关于添加打印命令的额外延迟的担忧,请记住,您不必在每次宏调用时都打印时间戳。相反,让宏调用获取时间戳并将其粘贴到列表中。程序完成后,打印列表中的所有时间戳。这样,您可以保存所有与打印相关的延迟,直到测试结束。

于 2010-10-21T17:40:53.860 回答
0

这是一些快速而肮脏的 C# 代码。基本上只是原始文件 IO 的东西。这不是很好,但我确实在大约 3 分钟内完成了它。这段代码暗示功能块用开头的注释行“//FunctionStart”和结尾的注释行“//FunctionEnd”来划分。有更优雅的方法可以做到这一点,这是快速/肮脏/骇客的方法。

使用托管应用程序来完成这项任务可能有点过头了,但是您可以通过简单地添加到这个功能来做很多自定义的东西。

        private void InsertMacro(string filePath)
        {
            //Declrations:
            StreamReader sr = new StreamReader(filePath);
            StreamWriter sw = new StreamWriter(filePath + ".tmp");
            string line = "";
            bool validBlock = false;

            //Go through source file line by line:
            while ((line = sr.ReadLine()) != null)
            {
                if (line == "//FunctionStart")
                    validBlock = true;
                else if (line == "//FunctionEnd")
                    validBlock = false;


                sw.WriteLine(line);

                if (validBlock)
                   sw.WriteLine("MYMACRO");
            }

            //Replace legacy source with updated source:
            File.Delete(filePath);
            File.Move(filePath + ".tmp", filePath);

            //Clean up streams:
            sw.Close();
            sr.Close();
        }
于 2010-10-21T18:04:56.453 回答
0

重写您的来源,以便以下工作:-)

而不是gcc ... file1.c file2.c ...

gcc ... `sed -e's/;/;\nMYMACRO/' file1.c` file1extra.c \
        `sed -e's/;/;\nMYMACRO/' file2.c` file2extra.c \
    ...
于 2010-10-21T17:48:18.150 回答