TCC(Tiny C Compiler)用C语言当脚本例子


        今天在我的C代码里面使用tcc当脚本,感觉比lua还爽, lua已经够小巧够精制了, Tiny C Compiler 更小, 速度也比lua要快, 关键是还能直接传指针参数或指针的返回值,就和C的函数调用一样方便,用TCC的好处是,我们可以把不变的代码用C来实现,把经常要改变的逻辑等用TCC来实现,这样以后修改逻辑相关的东西就不需要重新编译程序,用TCC最方便的地方就是直接把指针当参数和返回值在脚本和宿主程序之间传来传去。


下面是官方源代码里的一个例子 :

/*
 * Simple Test program for libtcc
 *
 * libtcc can be useful to use tcc as a "backend" for a code generator.
 */
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

#include "libtcc.h"

/* this function is called by the generated code */
int add(int a, int b)
{
    return a + b;
}

char my_program[] =
"int fib(int n)\n"
"{\n"
"    if (n <= 2)\n"
"        return 1;\n"
"    else\n"
"        return fib(n-1) + fib(n-2);\n"
"}\n"
"\n"
"int foo(int n)\n"
"{\n"
"    printf(\"Hello World!\\n\");\n"
"    printf(\"fib(%d) = %d\\n\", n, fib(n));\n"
"    printf(\"add(%d, %d) = %d\\n\", n, 2 * n, add(n, 2 * n));\n"
"    return 0;\n"
"}\n";

int main(int argc, char **argv)
{
    TCCState *s;
    int (*func)(int);

    s = tcc_new();
    if (!s) {
        fprintf(stderr, "Could not create tcc state\n");
        exit(1);
    }

    /* if tcclib.h and libtcc1.a are not installed, where can we find them */
    if (argc == 2 && !memcmp(argv[1], "lib_path=",9))
        tcc_set_lib_path(s, argv[1]+9);

    /* MUST BE CALLED before any compilation */
    tcc_set_output_type(s, TCC_OUTPUT_MEMORY);

    if (tcc_compile_string(s, my_program) == -1)
        return 1;

    /* as a test, we add a symbol that the compiled program can use.
       You may also open a dll with tcc_add_dll() and use symbols from that */
    tcc_add_symbol(s, "add", add);

    /* relocate the code */
    if (tcc_relocate(s, TCC_RELOCATE_AUTO) < 0)
        return 1;

    /* get entry symbol */
    func = tcc_get_symbol(s, "foo");
    if (!func)
        return 1;

    /* run the code */
    func(32);

    /* delete the state */
    tcc_delete(s);

    return 0;
}



下面的代码是我在自己的程序里面使用的例子

   

    int offset = 0;
    MatchResult_t Res;
    memset (&Res, 0, sizeof(Res));

    MatchField_t* pmf = pbq->arrMatch[idx];


    TCCState *script;

    script = tcc_new();
    if (!script) 
    {
        writelog(LOG_ERROR, "Could not create tcc state\n");
        return;
    }

    tcc_set_output_type(script, TCC_OUTPUT_MEMORY);

    //编译脚本文件
    if (tcc_add_file(script, "./script/hit.c") == -1)
    {
	writelog (LOG_ERROR, "Compiling error");
	tcc_delete(script);
	return;
    }

    //添加内部符号给脚本使用
    //tcc_add_symbol(s, "add", add);

    // relocate the code
    if (tcc_relocate(script, TCC_RELOCATE_AUTO) < 0)
    {
	tcc_delete(script);
	return;
    }
    
    //定义函数指针
    int (*func)(MatchField_t*, MatchResult_t*, char*, int);

    //获取函数地址
    func = tcc_get_symbol(script, "hitMatch");
    if (!func)
    {
	tcc_delete(script);
	return;
    }

    //通过函数地址 调用函数
    int ret = func(pmf, &Res, vbuf, vsiz);
    if(ret == 0)
    {
        //TODO...
    }
    tcc_delete(script);

脚本 hit.c 代码如下 :

int hitMatch (MatchField_t* pmf, MatchResult_t* pcr, char* vbuf, int vsiz)
{
    memcpy (pcr->dataType, vbuf, sizeof(int));
    //TODO....
    return 0;
}


  • 5
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
seata(Simple Extensible Autonomous Transaction Architecture)是一种开源的分布式事务解决方案,它提供了基于TCC(Try-Confirm-Cancel)的分布式事务处理模式。 假设我们有一个电商系统,其中涉及订单和库存两个服务。订单服务负责创建订单并扣除相应商品的库存,而库存服务则负责记录商品的库存数量。在这种情况下,如果订单服务成功创建了订单但库存服务因为某种原因失败了,有可能会导致订单和库存不一致。 为了解决这个问题,可以使用seata的TCC模式。首先,订单服务在尝试创建订单时会对库存服务发送“尝试扣除库存”的请求(Try)。如果库存服务成功扣除了库存,则订单服务会确认订单(Confirm);如果库存服务失败了,则订单服务会取消订单(Cancel)。 具体来说,在订单服务中,会有一个“Try”方法用于尝试创建订单和调用库存服务的“Try”方法;如果“Try”成功,则会有一个“Confirm”方法用于确认订单的创建;如果“Try”失败,则会有一个“Cancel”方法用于取消订单的创建。在库存服务中也类似地定义了“Try”、“Confirm”和“Cancel”方法。 通过使用seata的TCC模式,我们可以保证订单和库存的一致性,即使在订单服务和库存服务之间出现了故障或错误。这样,我们就可以避免因为分布式事务导致的数据不一致问题,提高了系统的可靠性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

langeldep

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值