<2024-01-07 周日>
emacs源码分析(七)
这DEFUN宏就像胶水一样,它把c代码和emacs-lisp代码给联系起来。但是DEFUN宏看着怪恐怖的有没有!
/* This version of DEFUN declares a function prototype with the right
arguments, so we can catch errors with maxargs at compile-time. */
#define DEFUN(lname, fnname, sname, minargs, maxargs, intspec, doc) \
SUBR_SECTION_ATTRIBUTE \
static union Aligned_Lisp_Subr sname = \
{
{
{
PVEC_SUBR << PSEUDOVECTOR_AREA_BITS }, \
{
.a ## maxargs = fnname }, \
minargs, maxargs, lname, {
intspec}, 0}}; \
Lisp_Object fnname
自己动手把emacs的DEFUN宏抠出来
为了方便理解,我把DEFUN宏给抠了出来,放在一个单独的工程里:ysouyno/t_emacs_defun,如果不想下载工程,本篇结尾会附上所有源码(仅一个文件,不到300行代码)。
关于这个工程要注意:
- 仅适用于
windows平台,为了编译方便,很多辅助宏能省略则省略。 - 设置
C++ Language Standard为ISO C++20 Standard (/std:c++20)。
挑了一个最简单的emacs-lisp函数eq:
DEFUN ("eq", Feq, Seq, 2, 2, 0,
doc: /* Return t if the two args are the same Lisp object. */
attributes: const)
(Lisp_Object obj1, Lisp_Object obj2)
{
if (EQ (obj1, obj2))
return Qt;
return Qnil;
}
展开后的eq代码是:
static union Aligned_Lisp_Subr Seq =
{
{
{
PVEC_SUBR << PSEUDOVECTOR_AREA_BITS }, // struct Lisp_Subr::header
{
.a2 = Feq }, // struct Lisp_Subr::function
2, // struct Lisp_Subr::min_args
2, // struct Lisp_Subr::max_args
"eq", // struct Lisp_Subr::symbol_name
{
0}, // struct Lisp_Subr::intspec
0 // struct Lisp_Subr::doc
}
};
Lisp_Object Feq(Lisp_Object obj1, Lisp_Object obj2)
{
if (EQ(obj1, obj2))
return Qt;
return Qnil;
}
从上可得,DEFUN有两个任务以(eq函数为例):
- 声明一个静态变量
Seq,它应该会将要用于emacs-lisp代码中的某些地方,目前我还不清楚细节。 - 定义一个
c函数Feq。
我照着DEFUN宏展开后样子定义了一个没有使用DEFUN宏来定义的函数my-eq,它可以正常工作:
// DEFUN("my-eq", ...)
static union Aligned_Lisp_Subr Smy_eq =
{
{
{
PVEC_SUBR << PSEUDOVECTOR_AREA_BITS },
{
.a2 = Fmy_eq },
2, 2, "my-eq", {
0}, 0} };
Lisp_Object Fmy_eq(Lisp_Object obj1, Lisp_Object obj2)
{
if (EQ(obj1, obj2))
return Qt;
return Qnil;
}
备注:
- 有一个
EXFUN宏要关注一下,这个代码我也抠出来了,它是用于声明Feq,否则编译器要报怨的:
error C2065: 'Feq': undeclared identifier
- 在
emacs源代码中,大量EXFUN的函数声明在globals.h
剖析Emacs源码:理解并实现DEFUN宏

最低0.47元/天 解锁文章
679

被折叠的 条评论
为什么被折叠?



