【lua学习】5.栈和lua_State

1 背景

  • lua使用的是基于“寄存器”的虚拟机实现方式,操作数放在指定索引的“寄存器”中,通过指令来访问这些“寄存器”实现操作。
    所谓“寄存器”是通过是通过栈结构实现的,而栈可以通过数字索引来操作指定的位置,因此非常符合寄存器的操作方式。
  • lua使用这种虚拟机的实现方式,最大的原因就是尽可能高效。
  • lua虚拟机运作,就伴随着数据的出栈和入栈。

2 栈(寄存器数组),虚拟机,全局状态机

2.1 栈定义在lua_State结构体中

每个lua虚拟机实例对应一个lua_State结构体,栈就定义在其中。

(lstate.h) lua_State

struct lua_State
{
	CommonHeader;//#define CommonHeader	GCObject *next; lu_byte tt; lu_byte marked
	lu_byte status;//虚拟机的错误状态码
	StkId top;//栈顶元素所在位置的下一个位置,也就是栈上第一个空闲的位置
	StkId base;//栈上,当前函数的基址(注意不是函数所在位置)
	global_State* l_G;//全局表,环境章节再说
	CallInfo* ci;//当前函数调用信息
	const Instruction* savedpc;//当前函数的指令位置,指向待取指指令的地址
	StkId stack_last;//栈最后一个位置的下一个位置
	StkId stack;//寄存器数组的起始位置
	CallInfo* end_ci;//函数调用信息数组的最后一个位置的下一个位置
	CallInfo* base_ci;//函数调用信息数组首地址
	int stacksize;//栈的大小
	int size_ci;//函数调用信息数组大小
	unsigned short nCcalls;//内嵌C调用层数
	unsigned short baseCcalls;//唤醒协程时的内嵌C调用层数
	lu_byte hookmask;
	lu_byte allowhook;
	int basehookcount;
	int hookcount;
	lua_Hook hook;
	TValue l_gt;//Global表
	TValue env;//环境表的临时位置
	GCObject* openupval;//open状态的upvalues
	GCObject* gclist;
	struct lua_longjmp* errorJmp;//跳转信息单链表,实现try catch的功能,见函数章节
	ptrdiff_t errfunc;//当前错误处理函数在栈上的索引
};

lua_State使用TValue数组来模拟栈,包括几个重要的数据成员

  • stack 寄存器数组的起始位置
  • base 当前函数栈的基地址(一般是第一个参数的地址)(每个函数各自的栈并不是新起的一个栈,而是对应于lua_State栈上的某段空间)
  • top 当前函数栈的下一个可用位置
  • stack_last 数组的最后一个可用位置
  • stacksize 栈数组的大小

2.2 global_State 全局状态机

(lstate.h) global_State

typedef struct global_State
{
	stringtable strt;//全局字符串表
	lua_Alloc freealloc;//内存重分配函数
	void* ud;//freealloc的辅助数据
	lu_byte currentwhite;//当前白色,见GC章节
	lu_byte gcstate;//GC状态,见GC章节
	int sweepstrgc;//strt中GC扫描到的位置
	GCObject* rootgc;//所有可回收对象的链表
	GCObject** sweepgc;//rootgc中扫描到的位置
	GCObject* gray;//灰色对象链表
	GCObject* grayagain;//需要被原子性遍历地对象链表
	GCObject* week;//弱表的链表
	GCObject* tmudata;//需要被GC的userdata的链表的最后一个元素
	Mbuffer* buff;//字符串连接操作用的临时缓冲对象
	lu_mem GCthreshold;//触发GC的边界值,见GC章节
	lu_mem totalbytes;//当前分配的总字节数
	lu_mem estimate;//实际上使用的总字节数的估计值
	lu_mem gcdept;//在预定的回收字节数中,还欠多少字节没有回收
	int gcpause;//连续的GC中的停顿步长相关值,见GC章节
	int gcstepmul;//GC的粒度
	lua_CFunction panic;//对于未捕获异常,会调用这个函数
	TValue l_registry;//注册表
	struct lua_State* mainthread;//主虚拟机(主协程)
	UpVal uvhead;//open状态的upvalue双链表的头部
	struct Table* mt[NUM_TAGS];//存放 基本类型的元表 的数组
	TString* tmname[TM_N];//存放 元方法名字符串对象地址 的数组
}

2.3 lua_newstate 主虚拟机和全局状态机的创建

(lstate.c) lua_newstate 虚拟机创建

//typedef void * (*lua_Alloc) (void *ud, void *ptr, size_t osize, size_t nsize); 内存分配函数类型
LUA_API lua_State* lua_newstate(lua_Alloc f, void* ud)
{
	//#define LUAI_EXTRASPACE 0
	//#define state_size(x) (sizeof(x) + LUAI_EXTRASPACE)
	void* l = (*f)(ud, NULL, 0, state_size(LG));//分配主虚拟机和全局状态机的内存
	if (l == NULL)
	{
		return NULL;
	}
	//#define tostate(l) (cast(lua_State *, cast(lu_byte *, l) + LUAI_EXTRASPACE))
	lua_State* L = tostate(l);
	global_State* g = &((LG*)L)->g;
	L->next = NULL;
	L->tt = LUA_TTHREAD;
	//#define bitmask(b) (1<<(b))
	//#define bit2mask(b1,b2) (bitmask(b1) | bitmask(b2))
	g->currentwhite = bit2mask(WHITE0BIT, FIXEDBIT);
	//#define WHITEBITS	bit2mask(WHITE0BIT, WHITE1BIT)
	//#define luaC_white(g)	cast(lu_byte, (g)->currentwhite & WHITEBITS)
	L->marked = luaC_white(g);
	//#define setbits(x,m)	((x) |= (m))
	//#define set2bits(x,b1,b2)	setbits(x, (bit2mask(b1, b2)))
	set2bits(L->marked, FIXDEBIT, SFIXEDBIT);
	preinit_state(L, g);
	g->frealloc = f;
	g->ud = ud;
	g->mainthread = L;
	g->uvhead.u.l.prev = &g->uvhead;
	g->uvhead.u.l.next = &g->uvhead;
	g->GCthreshold = 0;
	g->strt.size = 0;
	g->strt.nuse = 0;
	g->strt.hash = NULL;
	setnilvalue(registry(L));
	luaZ_initbuffer(L, &g->buff);//#define luaZ_initbuffer(L, buff) ((buff)->buffer = NULL, (buff)->buffsize = 0)
	g->panic = NULL;
	g->gcstate = GCSpasuse;
	g->rootgc = obj2gco(L);
	g->sweepstrgc = 0;
	g->sweepgc = &g->rootgc;
	g->gray = NULL;
	g->grayagain = NULL;
	g->weak = NULL;
	g->tmudata = NULL;
	g->totalbytes = sizeof(LG);
	g->gcpause = LUAI_GCPAUSE;//#define LUAI_GCPAUSE 200  /* 200% (wait memory to double before next GC) */
	g->gcstepmul = LUAI_GCMUL;//#define LUAI_GCMUL 200 /* GC runs 'twice the speed' of memory allocation */
	g->gcdebt = 0;
	for (int i = 0; i < NUM_TAGS; i++)
	{
		g->mt[i] = NULL;
	}
	if (luaD_rawrunprotected(L, f_luaopen, NULL) != 0)//luaD_rawrunprotected见函数章节
	{
		close_state(L);
		L = NULL;
	}
	return L;
}

(lstate.c) LG 主虚拟机和全局状态机的组合结构

typedef struct LG {
  lua_State l;
  global_State g;
} LG;

(lstate.c) preinit_state 初始化虚拟机/协程

static void preinit_state(lua_State* L, global_State* g)
{
	G(L) = g;
	L->stack = NULL;
	L->stacksize = 0;
	L->errorJmp = NULL;
	L->hook = NULL;
	L->hookmask = 0;
	L->basehookcount = 0;
	L->allowhook = 1;
	resethookcount(L);//#define resethookcount(L)	(L->hookcount = L->basehookcount)
	L->openupval = NULL;
	L->size_ci = 0;
	L->cCcalls = L->baseCcalls = 0;
	L->status = 0;
	L->base_ci = L->ci = NULL;
	L->savedpc = NULL;
	L->errfunc = 0;
	setnilvalue(gt(L));
}

(lstate.c) f_luaopen 开启虚拟机

static void f_luaopen(lua_State* L, void* ud)
{
	global_State* g = G(L);
	//栈初始化
	stack_init(L, L);
	//新建Global表
	sethvalue(L, gt(L), luaH_new(L, 0, 2));
	//新建注册表
	sethvalue(L, registry(L), luaH_new(L, 0, 2));
	//初始化全局字符串表
	luaS_resize(L, MINSTRTABSIZE);//#define MINSTRTABSIZE 32
	//元方法名初始化
	luaT_init(L);
	//初始化保留字
	luaX_init(L);
	//#define l_setbit(x,b)	setbits(x, bitmask(b))
	//#define luaS_fix(s)	l_setbit((s)->tsv.marked, FIXEDBIT)
	//#define MEMERRMSG	"not enough memory"
	luaS_fix(luaS_newliteral(L, MEMERRMSG));
	g->GCthreashold = 4 * g->totalbytes;
}

(lstate.c) stack_init 栈初始化

static void stack_init(lua_State* L1, lua_State* L2)
{
	//初始化 CallInfo 数组
	L1->base_ci = luaM_newvector(L, BASIC_CI_SIZE, CallInfo);//#define BASIC_CI_SIZE 8
	L1->ci = L1->base_ci;
	L1->size_ci = BASIC_CI_SIZE;
	L1->end_ci = L1->base_ci + L1->size_ci - 1;
	//初始化 “寄存器” 数组
	L1->stack = luaM_newvector(L, BASIC_STACK_SIZE + EXTRA_STACK, TValue);
	L1->stacksize = BASIC_STACK_SIZE + EXTRA_STACK;
	L1->top = L1->stack;
	L1->stack_last = L1->stack + (L1->stacksize - EXTRA_STACK) - 1;
	//初始化第一个CallInfo
	L1->ci->func = L1->top;
	setnilvalue(L1->top++);
	L1->base = L1->ci->base = L1->top;
	L1->ci->top = L1->top + LUA_MINSTACK;//#define LUA_MINSTACK	20
}

(ltm.c) luaT_init 元方法名初始化

void luaT_init(lua_State* L)
{
	static const char* const luaT_eventname[] = {
		"__index", "__newindex", "__gc", "__mode", "__eq", "__add", "__sub", "__mul", "__div", "__mod",
		"__pow", "__unm", "__len", "__lt", "__le", "__concat", "__call"
	};
	for (int i = 0; i < TM_N; i++)
	{
		G(L)->tmname[i] = luaS_new(L, luaT_eventname[i]);
		luaS_fix(G(L)->tmname[i]);
	}
}

(ltm.h) TM_N 元方法名的数量

typedef enum {
  TM_INDEX,
  TM_NEWINDEX,
  TM_GC,
  TM_MODE,
  TM_EQ,  /* last tag method with `fast' access */
  TM_ADD,
  TM_SUB,
  TM_MUL,
  TM_DIV,
  TM_MOD,
  TM_POW,
  TM_UNM,
  TM_LEN,
  TM_LT,
  TM_LE,
  TM_CONCAT,
  TM_CALL,
  TM_N		/* number of elements in the enum */
} TMS;

(llex.c) luaX_init 初始化保留字

void luaX_init(lua_State* L)
{
	for (int i = 0; i < NUM_RESERVED; i++)//#define NUM_RESERVED (cast(int, TK_WHILE-FIRST_RESERVED+1))
	{
		//尝试新建每个保留字字符串
		TString* ts = luaS_new(L, luaX_tokens[i]);
		//标记不会被GC,修改ts->tsv.marked为FIXEDBIT
		luaS_fix(ts);
		//记录在保留字数组的索引+1值
		ts->tsv.reserved = cast_byte(i + 1);
	}
}

(llex.h) 保留字枚举

enum RESERVED {
  /* terminal symbols denoted by reserved words */
  TK_AND = FIRST_RESERVED, TK_BREAK,
  TK_DO, TK_ELSE, TK_ELSEIF, TK_END, TK_FALSE, TK_FOR, TK_FUNCTION,
  TK_IF, TK_IN, TK_LOCAL, TK_NIL, TK_NOT, TK_OR, TK_REPEAT,
  TK_RETURN, TK_THEN, TK_TRUE, TK_UNTIL, TK_WHILE,
  /* other terminal symbols */
  TK_CONCAT, TK_DOTS, TK_EQ, TK_GE, TK_LE, TK_NE, TK_NUMBER,
  TK_NAME, TK_STRING, TK_EOS
};

(llex.c) luaX_tokens 保留字的数组

const char *const luaX_tokens [] = {
    "and", "break", "do", "else", "elseif",
    "end", "false", "for", "function", "if",
    "in", "local", "nil", "not", "or", "repeat",
    "return", "then", "true", "until", "while",
    "..", "...", "==", ">=", "<=", "~=",
    "<number>", "<name>", "<string>", "<eof>",
    NULL
};

2.4 lua_close 关闭虚拟机

(lstate.c) lua_close

LUA_API void lua_close(lua_State* L)
{
	L = G(L)->mainthread;//只有主虚拟机才可以被关闭
	//关闭所有的upvalue
	luaF_close(L, L->stack);//luaF_close见函数章节
	//分离那些有GC元方法的userdata,见GC章节
	luaC_separateudata(L, 1);
	L->errfunc = 0;
	do 
	{
		L->ci = L->base_ci;
		L->base = L->top = L->ci->base;
		L->nCcalls = L->baseCcalls = 0;
	} while (luaD_rawrunprotected(L, callallgcTM, NULL) != 0);//见callallgcTM章节
	close_state(L);
}

(lstate.c) close_state 关闭虚拟机的收尾工作

static void close_state(lua_State* L)
{
	global_State* g = G(L);
	luaF_close(L, L->stack);
	//回收所有的对象
	luaC_freeall(L);//luaC_freeall见GC章节
	luaM_freearray(L, G(L)->strt.hash, G(L)->strt.size, TString*);
	luaZ_freebuffer(L, &g->buff);
	freestack(L, L);
	//#define fromstate(l) (cast(lu_byte *, (l)) - LUAI_EXTRASPACE)
	(*g->frealloc)(g->ud, fromstate(L), state_size(LG), 0);
}

(lstate.c) freestack 释放协程L1上“寄存器”数组和CallInfo数组

static void freestack(lua_State* L, lua_State* L1)
{
	luaM_freearray(L, L1->base_ci, L1->size_ci, CallInfo);
	luaM_freearray(L, L1->stack, L1->stacksize, TValue);
}

3 栈上的地址

虽然可根据数字索引定位栈中地地址,但是官方也定义了几个不在栈中的假索引

3.1 假索引

(lua.h)

#define LUA_REGISTRYINDEX	(-10000) //注册表
#define LUA_ENVIRONINDEX	(-10001) //环境表
#define LUA_GLOBALSINDEX	(-10002) //全局变量表
#define lua_upvalueindex(i)	(LUA_GLOBALSINDEX-(i)) //C函数upvalue表

3.2 根据数字索引获取栈上的地址

(lapi.c) index2adr

static TValue* index2adr(lua_State* L, int idx)
{
	//若 idx > 0,则是以L->base为基准,要找的地址时 L->base + idx - 1
	if (idx > 0)
	{
		TValue* o = L->base + idx - 1;//因为base就是相对的1,所以要减1啦
		//越上界了,当然返回nil啦
		if (o >= L->top)
		{
			return cast(TValue*, luaO_nilobject);
		}
		//没越界,就返回找到的对象地址
		return o;
	}
	//若 LUA_REGISTRYINDEX < idx <= 0,则是以 L->top 为基准,要找的地址时 L->top + idx
	if (idx > LUA_REGISTRYINDEX)
	{
		//以栈顶为相对位置返回对象地址
		return L->top + idx;
	}
	
	//不满足以上取值范围的idx都是假索引,因为不是从栈里取地址了,而是特殊处理,如下文...
	
	//若 假索引 为 注册表索引,就返回注册表地址
	if (idx == LUAREGISTRYINDEX)
	{
		return registry(L);//#define registry(L) (&G(L)->l_registry)
	}
	//若 假索引 为 环境表索引,则返回当前函数的env表
	if (idx == LUA_ENVIRONINDEX)
	{
		//获取当前函数环境的闭包地址
		Closure* func = curr_func(L);//#define curr_func(L)	(clvalue(L->ci->func))
		//设置当前函数环境为当前lua虚拟机的环境
		sethvalue(L, &L->env, func->c.env);
		return &L->env;
	}
	//若 假索引 为 全局表索引,则返回当前lua_State的全局表
	if (idx == LUA_GLOBALSINDEX)
	{
		return gt(L);//#define gt(L) (&L->l_gt)
	}
	//除以上的所有情况(也就是从-10003到负无穷),就是访问C函数的upvalue了 
	Closure* func = curr_func(L);
	//计算upvalue的索引
	idx = lua_upvalueindex(idx);
	//返回upvalue地址或者nil
	return (idx <= func->c.nupvalues) ? &func->c.upvalue[idx - 1] : cast(TValue*, luaO_nilobject);
}

4 基本的栈操作API

4.1 lua_gettop 获取 栈顶 相对于 当前函数栈基址 的偏移

(lapi.c) lua_gettop

LUA_API int lua_gettop(lua_State* L)
{
	return cast_int(L->top - L->base);
}

4.2 lua_settop 设置栈顶的位置

(lapi.c) lua_settop

LUA_API int lua_settop(lua_State* L, int idx)
{
	if (idx >= 0)
	{
		while (L->top < L->base + idx)
		{
			setnilvalue(L->top++);
		}
		L->top = L->base + idx;
	}
	else
	{
		L->top += idx + 1;
	}
}

4.3 lua_pushvalue 将指定索引的值复制到栈顶

(lapi.c) lua_pushvalue

LUA_API void lua_pushvalue(lua_State* L, int idx)
{
	setobj2s(L, L->top, index2adr(L, idx));
	api_incr_top(L);//#define api_incr_top(L)   {api_check(L, L->top < L->ci->top); L->top++;}
}

4.4 lua_remove 将指定索引的值入栈,且上面的值依次下移一位

(lapi.c) lua_remove

LUA_API void lua_remove(lua_State* L, int idx)
{
	StkId p = index2adr(L, idx);
	while (++p < L->top)
	{
		setobjs2s(L, p - 1, p);
	}
	L->top--;
}

4.5 lua_insert 将栈顶的值插入到指定索引,且上面的值依次上移一位

(lapi.c) lua_insert

LUA_API void lua_insert(lua_State* L, int idx)
{
	StkId p = index2adr(L, idx);
	for (StkId q = L->top; q > p; q--)
	{
		setobjs2s(L, q, q - 1);
	}
	setobjs2s(L, p, L->top);
}

4.6 lua_replace 将栈顶的值替代掉指定索引的值

(lapi.c) lua_replace

LUA_API void lua_replace(lua_State* L, int idx)
{
	//若想用栈顶的值替换 base_ci的环境表,则报错
	if (idx = LUA_ENVIRONINDEX && L->ci == L->base_ci)
	{
		luaG_runerror(L, "no calling environment");
	}
	StkId o = index2adr(L, idx);
	if (idx == LUA_ENVIRONINDEX)
	{
		Closure* func = curr_func(L);//curr_func见环境章节
		func->c.env = hvalue(L->top - 1);
		luaC_barrier(L, func, L->top - 1);//luaC_barrier见GC章节
	}
	else
	{
		setobj(L, o, L->top - 1);
		if (idx < LUA_GLOBALSINDEX)
		{
			luaC_barrier(L, curr_func(L), L->top - 1);
		}
	}
	L->top--;
}

4.7 lua_checkstack 检查栈可用空间是否够size个,不够的话且没超过范围,则扩容。返回值若返回非0则表示扩容成功

(lapi.c) lua_checkstack

LUA_API int lua_checkstack(lua_State* L, int size)
{
	int res = 1;
	//#define LUAI_MAXCSTACK 8000
	if (size > LUAI_MAXCSTACK || (L->top - L->base + size) > LUAI_MAXCSTACK)
	{
		res = 0;
	}
	else if (size > 0)
	{
		//luaD_checkstack 若栈空间不足size个,则扩容
		/*
			#define luaD_checkstack(L,n) \
  				if ((char *)L->stack_last - (char *)L->top <= (n)*(int)sizeof(TValue)) \
    				luaD_growstack(L, n); \
  				else condhardstacktests(luaD_reallocstack(L, L->stacksize - EXTRA_STACK - 1));
		*/
		luaD_checkstack(L, size);
		if (L->ci->top < L->top + size)
		{
			L->ci->top = L->top + size;
		}
	}
	return res;
}

4.8 lua_xmove 将协程from的n个值从栈拿出,以原顺序压入协程to的栈中

(lapi.c) lua_xmove

LUA_API void lua_xmove(lua_State* from, lua_State* to, int n)
{
	if (from == to) 
	{
		return;
	}
	from->top -= n;
	for (int i = 0; i < n; i++)
	{
		setobj2s(to, to->top++, from->top + i);
	}
}

5 C访问栈

5.1 lua_isnumber 判断指定索引的值是否是数字

(lapi.c) lua_isnumber

LUA_API int lua_isnumber(lua_State* L, int idx)
{
	TValue n;
	const TValue* o = index2adr(L, idx);
	return tonumber(o, &n);
}

(lvm.h) tonumber

#define tonumber(o,n)	(ttype(o) == LUA_TNUMBER || (((o) = luaV_tonumber(o,n)) != NULL))

(lvm.c) luaV_tonumber 尝试将obj转为数字,在地址n输出。(若obj就是数字类型=>返回obj地址;若obj是字符串类型且执行转化为数字成功,则返回n地址;否则返回NULL)

const TValue* luaV_tonumber(const TValue* obj, TValue* n)
{
	lua_Number num;
	if (ttisnumber(obj))//#define ttisnumber(o)	(ttype(o) == LUA_TNUMBER)
	{
		return obj;
	}
	//#define ttisstring(o)	(ttype(o) == LUA_TSTRING)
	//#define svalue(o)       getstr(rawtsvalue(o))
	//#define rawtsvalue(o)	check_exp(ttisstring(o), &(o)->value.gc->ts)
	//#define getstr(ts)	cast(const char *, (ts) + 1)
	if (ttisstring(obj) && luaO_str2d(svalue(obj), &num))
	{
		setnvalue(n, num);
		return n;
	}
	return NULL;
}

(lobject.c) luaO_str2d 尝试将字符串转为数字,以result为输出地址(返回值1表示成功,0表示失败)

int luaO_str2d(const char* s, lua_Number* result)
{
	char* endptr;
	*result = lua_str2number(s, &endptr);//#define lua_str2number(s,p)	strtod((s), (p))
	if (endptr == s)//转换失败
	{
		return 0;
	}
	if (*endptr == 'x' || *endptr == 'X')//十六进制数
	{
		*result = cast_num(strtoul(s, &endptr, 16));
	}
	if (*endptr == '\0')
	{
		return 1;
	}
	while (isspace(cast(unsigned char, *endptr)))//跳过空白字符
	{
		endptr++;
	}
	if (*endptr != '\0')
	{
		return 0;
	}
	return 1;
}

5.2 lua_isstring判断指定索引的值是否是字符串(字符串和数字都算字符串)

(lapi.c) lua_isstring

LUA_API int lua_isstring(lua_State* L, int idx)
{
	int t = lua_type(L, idx);
	return (t == LUA_TSTRING || t == LUA_TNUMBER);
}

5.3 lua_iscfunction 判断指定索引的值是否C函数

(lapi.c) lua_iscfunction

LUA_API int luac_iscfunction(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	return iscfunction(o);//#define iscfunction(o)	(ttype(o) == LUA_TFUNCTION && clvalue(o)->c.isC)
}

5.4 lua_isuserdata 判断指定索引的值是否是userdata类型

(lapi.c) lua_isuserdata

LUA_API int lua_isuserdata(lua_State* L, int idx)
{
	const TValue* o = index2adr(L, idx);
	//#define ttisuserdata(o)	(ttype(o) == LUA_TUSERDATA)
	//#define ttislightuserdata(o)	(ttype(o) == LUA_TLIGHTUSERDATA)
	return (ttisuserdata(o) || ttislightuserdata(o));
}

5.5 lua_type 获取指定索引值的类型

(lapi.c) lua_type

LUA_API int lua_type(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	return (o == luaO_nilobject) ? LUA_TNONE : ttype(o);//#define ttype(o)	((o)->tt)
}

5.6 lua_typename 获取指定类型的类型名

(lapi.c) lua_typename

LUA_API const char* lua_typename(lua_State* L, int t)
{
	return (t == LUA_TNONE) ? "no value" : luaT_tynames[t];
}

(ltm.c) luaT_typenames

const char *const luaT_typenames[] = {
  "nil", "boolean", "userdata", "number",
  "string", "table", "function", "userdata", "thread",
  "proto", "upval"
};

5.7 lua_equal 判断2个索引对应的值是否相等

(lapi.c) lua_equal

LUA_API int lua_equal(lua_State* L, int index1, int index2)
{
	StkId o1 = index2adr(L, index1);
	StkId o2 = index2adr(L, index2);
	//只要有一个为luaO_nilobject 则必定不相等
	//#define equalobj(L,o1,o2) (ttype(o1) == ttype(o2) && luaV_equalval(L, o1, o2))
	int i = (o1 == luaO_nilobject || o2 == luaO_nilobject) ? 0 : equalobj(L, o1, o2);
	return i;
}

(lvm.c) luaV_equalval 判断俩类型相同的值是否相等(注意:大前提是类型相同)

int luaV_equalval(lua_State* L, const TValue* t1, const TValue* t2)
{
	const TValue* tm;
	switch(ttype(t1))
	{
		case LUA_TNIL:
		{
			return 1;
		}
		case LUA_TNUMBER:
		{
			//#define luai_numeq(a,b)		((a)==(b))
			//#define nvalue(o)	check_exp(ttisnumber(o), (o)->value.n)
			return luai_numeq(nvalue(t1), nvalue(t2));
		}
		case LUA_TBOOLEAN:
		{
			//#define bvalue(o)	check_exp(ttisboolean(o), (o)->value.b)
			return bvalue(t1) == bvalue(t2);
		}
		case LUA_TLIGHTUSERDATA:
		{
			//#define pvalue(o)	check_exp(ttislightuserdata(o), (o)->value.p)
			return pvalue(t1) == pvalue(t2);
		}
		case LUA_TUSERDATA:
		{
			//#define uvalue(o)	(&rawuvalue(o)->uv)
			if (uvalue(t1) == uvalue(t2))
			{
				return 1;
			}
			tm = get_compTM(L, uvalue(t1)->metatable, uvalue(t2)->metatble, TM_EQ);
			break;
		}
		case LUA_TATBLE:
		{
			//#define hvalue(o)	check_exp(ttistable(o), &(o)->value.gc->h)
			if (hvalue(t1) == hvalue(t2)) 
			{
				return 1;
			}
			tm = get_compTM(L, hvalue(t1)->metatable, hvalue(t2)->metatble, TM_EQ);
			break;
		}
		default:
		{
			return gcvalue(t1) == gcvalue(t2);
		}
	}
	if (tm == NULL)
	{
		return 0;
	}
	callTMres(L, L->top, tm, t1, t2);
	//#define l_isfalse(o)	(ttisnil(o) || (ttisboolean(o) && bvalue(o) == 0))
	return !l_isfalse(L->top);
}

(lvm.c) get_compTM

static const TValue* get_compTM(lua_State* L, Table* mt1, Table* mt2, TMS event)
{
	const TValue* tm1 = fasttm(L, mt1, event);
	if (tm1 == NULL)
	{
		return NULL;
	}
	if (mt1 == mt2)
	{
		return tm1;//相同的元表,当然是相同的元方法啦
	}
	const TValue* tm2 = fasttm(L, mt2, event);
	if (tm2 == NULL)
	{
		return NULL;
	}
	if (luaO_rawequalObj(tm1, tm2))
	{
		return tm1;
	}
	return NULL;
}

(ltm.h) fasttm (含义:快速获取 元表et 关于 元方法标识e 的元方法)
快速在于:

  • 先判断 元表et 的flags 关于e的标志位 是否为真,若真则表示无该元方法,无需后面的查表过程
  • 否则,搜索global_State里的tmname数组获取元方法名,再调用luaT_gettm获取元方法
#define gfasttm(g,et,e) ((et) == NULL ? NULL : \
  ((et)->flags & (1u<<(e))) ? NULL : luaT_gettm(et, e, (g)->tmname[e]))
#define fasttm(l,et,e)	gfasttm(G(l), et, e)

(ltm.c) luaT_gettm (含义:根据 元方法名ename 获取 元表events 的元方法,若获取不到则根据元方法标识event设置 元方法events的flags)

const TValue* luaT_gettm(Table* events, TMS event, TString* ename)
{
	const TValue* tm = luaH_getstr(events, ename);
	//若没找到元方法,则标记这个元表的flags,以便下次访问的时候节省查找耗时
	if (ttisnil(tm))
	{
		events->flags |= cast_byte(1u << event);
		return NULL;
	}
	return tm;
}

(lobject.c) luaO_rawequalObj 直接判断2对象是否相等

int luaO_rawequalObj(const TValue* t1, const TValue* t2)
{
	//若类型都不相同,则就是不相同
	if (ttype(t1) != ttype(t2))
	{
		return 0;
	}
	switch (ttype(t1))
	{
		//nil类型只有一个值,必定相同
		case LUA_TNIL:
		{
			return 1;
		}
		//数字地话,判断t->value.n是否相同
		case LUA_TNUMBER:
		{
			return luai_numeq(nvalue(t1), nvalue(t2));
		}
		//若为bool类型,判断t->value.b是否相同
		case LUA_TBOOLEAN:
		{
			return bvalue(t1)==bvalue(t2);
		}
		//若为lightuserdata类型,判断t->value.p是否相等
		case LUA_TLIGHTUSERDATA:
		{
			return pvalue(t1)==pvalue(t2);
		}
		//否则判断t->value.gc是否指向同一个GCObject
		default:
		{
			return gcvalue(t1)==gcvalue(t2);
		}
	}
}

(lvm.c) callTMres (含义: res=f(p1, p2))

//@param res 返回值位置
//@param f 元方法
//@param p1 参数1
//@param p2 参数2
static void callTMres(lua_State* L, StkId res, const TValue* f, const TValue* p1, const TValue* p2)
{
	//记录返回值地址 和 栈基址 的差值
	ptrdiff_t result = savestack(L, res);//#define savestack(L,p)		((char *)(p) - (char *)L->stack)
	//元方法入栈
	setobj2s(L, L->top, f);
	//参数1入栈
	setobj2s(L, L->top+1, p1);
	//参数2入栈
	setobj2s(L, L->top+2, p2);
	//检查栈够不够三个元素,不够的话就扩容
	luaD_checkstack(L, 3);
	//入栈了3个元素,当然栈顶地址+3
	L->top += 3;
	//执行函数,函数到栈顶中的值都是参数,返回值是1个
	luaD_call(L, L->top - 3, 1);//luaD_call,见函数章节
	//根据偏移量获取返回值在栈上的地址(因为前面的luaD_checkstack可能会导致lua栈被重新分配地址)
	res = restorestack(L, result);//#define restorestack(L,n)	((TValue *)((char *)L->stack + (n)))
	//函数返回值出栈,将栈顶的值复制到返回值位置
	L->top--;
	setobj2s(L, res, L->top);
}

5.8 lua_rawequal判断2个索引对应的值是否相等(不通过元表)

(lapi.c) lua-rawequal

LUA_API int lua_rawequal(lua_State* L, int index1, int index2)
{
	StkId o1 = index2adr(L, index1);
	StkId o2 = index2adr(L, index2);
	return (o1 == luaO_nilobject || o2 == luaO_nilobject) ? 0 : luaO_rawequalObj(o1, o2);
}

5.9 lua_lessthan判断索引1的值是否比索引2的值小

(lapi.c) lua_lessthan

LUA_API int lua_lessthan(lua_State* L, int index1, int index2)
{
	StkId o1 = index2adr(L, index1);
	StkId o2 = index2adr(L, index2);
	return (o1 == luaO_nilobject || o2 == luaO_nilobject) ? 0 : luaV_lessthan(o1, o2);
}

(lvm.c) luaV_lessthan

int luaV_lessthan(lua_State* L, const TValue* l, const TValue* r)
{
	int res;
	if (ttype(l) != ttype(r))//类型不相等,则报错
	{
		return luaG_ordererror(L, l, r);//luaG_ordererror,见异常章节
	}
	else if (ttisnumber(l))//左边是数字,则双方都取数字比较
	{
		//#define luai_numlt(a,b) ((a)<(b))
		return luai_numlt(nvalue(l), nvalue(r));
	}
	else if (ttistring(l))
	{
		//#define rawtsvalue(o)	check_exp(ttisstring(o), &(o)->value.gc->ts)
		return l_strcmp(rawtsvalue(l), rawtsvalue(r)) < 0;
	}
	else if ((res = call_orderTM(L, l, r, TM_LT)) != -1)
	{
		return res;
	}
	return luaG_ordererror(L, l, r);
}

(lvm.c) l_strcmp 比较字符串

static int l_strcmp(const TString* ls, const TString* rs)
{
	const char* l = getstr(ls);//#define getstr(ts)	cast(const char *, (ts) + 1)
	size_t ll = ls->tsv.len;//l串的待比较长度
	const char* r = getstr(rs);
	size_t lr = rs->tsv.len;//r串的待比较长度
	for (;;)
	{
		int temp = strcoll(l, r);//l>r则>0,l==r则=0,l<r则<0
		if (temp != 0)
		{
			return temp;
		}
		else
		{
			size_t len = strlen(l);
			if (len == lr)//r串结束了
			{
				return (len == ll) ? 0 : 1;
			}
			else if (len == ll)//r串未结束,l串结束了
			{
				return -1;
			}
			len++;
			l += len;
			ll -= len;
			r += len;
			lr -= len;
		}
	}
}

(lvm.c) call_orderTM 执行元方法(前提:从p1获取的元方法要和从p2获取的元方法要相等)

static int call_orderTM(lua_State* L, const TValue* p1, const TValue* p2, TMS event)
{
	const TValue* tm1 = luaT_gettmbyobj(L, p1, event);
	if (ttisnil(tm1))
	{
		return -1;
	}
	const TValue* tm2 = luaT_gettmbyobj(L, p2, event);
	if (!luaO_rawequalObj(tm1, tm2))
	{
		return -1;
	}
	callTMres(L, L->top, tm1, p1, p2);
	return !is_false(L->top);
}

(ltm.c) luaT_gettmbyobj (根据 元方法标识event 获取 对象o 的元方法)

const TValue* luaT_gettmbyobj(lua_State* L, const TValue* o, TMS event)
{
	Table* mt;
	switch(ttype(o))
	{
		case LUA_TTABLE:
		{
			mt = hvalue(o)->metatable;
			break;
		}
		case LUA_TUSERDATA:
		{
			mt = uvalue(o)->metatble;
			break;
		}
		default:
		{
			mt = G(L)->mt[ttype(o)];//非table非userdata类型的元表统统在global_State的mt数组内
		}
	}
	//为何不用 fasttm(L, mt, event) 呢?
	return (mt ? luaH_getstr(mt, G(L)->tmname[event]) : luaO_nilobject);
}

5.10 lua_tonumber 以数字的方式解析指定索引的值(原值不变)

(lapi.c) lua_tonumber

LUA_API lua_Number lua_tonumber(lua_State* L, int idx)
{
	TValue n;
	const TValue* o = index2adr(L, idx);
	if (tonumber(o, &n))
	{
		return nvalue(o);
	}
	return 0;
}

5.11 lua_tointeger 以整数的方式解析指定索引的值(原值不变)

(lapi.c) lua_tointeger

LUA_API lua_Integer lua_tointeger(lua_State* L, int idx)
{
	TValue n;
	const TValue* o = index2adr(L, idx);
	if (tonumber(o, &n))
	{
		lua_Integer res;
		lua_Number num = nvalue(o);
		lua_number2integer(res, num);
		return res;
	}
	return 0;
}

5.12 lua_toboolean 以bool的方式解析指定索引的值(原值不变)

(lapi.c) lua_toboolean

LUA_API int lua_toboolean(lua_State* L, int idx)
{
	const TValue* o = index2adr(L, idx);
	return !l_isfalse(o);
}

5.13 lua_tolstring 以string的形式解析指定索引的值(会导致原值改变)

(lapi.c) lua_tolstring

LUA_API const char* lua_tolstring(lua_State* L, int idx, size_t* len)
{
	StkId o = index2adr(L, idx);
	if (!ttisstring(o))//#define ttisstring(o)	(ttype(o) == LUA_TSTRING)
	{
		if (!luaV_tostring(L, o))
		{
			if (len != NULL)
			{
				*len =0;
			}
			return NULL;
		}
		luaC_checkGC(L);
		o = index2adr(L, idx);//前面的操作可能回导致栈的重新分配,所以需要重新获取一下
	}
	if (len != NULL)
	{
		*len = tsvalue(o)->len;//#define tsvalue(o)	(&rawtsvalue(o)->tsv)
	}
	//#define getstr(ts)	cast(const char *, (ts) + 1)
	//#define svalue(o)       getstr(rawtsvalue(o))
	return svalue(o);
}

(lvm.c) luaV_tostring 将非字符串类型能否转为字符串类型,转成功则返回true

int luaV_tostring(lua_State* L, StkId obj)
{
	//不是数字,则不能
	if (!ttisnumber(obj))
	{
		return 0;
	}
	char s[LUAI_MAXNUMBER2STR];//#define LUAI_MAXNUMBER2STR	32
	lua_Number n = nvalue(obj);
	//#define LUA_NUMBER_FMT "%.14g" //系统自动选择占宽度较小的某种格式输出,g格式符不输出小数点后无意义的零
	//#define lua_number2str(s,n) sprintf((s), LUA_NUMBER_FMT, (n))
	lua_number2str(s, n);
	//把字符串赋值给obj位置
	setsvalue2s(L, obj, luaS_new(L, s));
	return 1;
}

5.14 lua_objlen 获取指定索引的值的长度(如果是数字,还会将原值转为字符串类型)

(lapi.c) lua_objlen

LUA_API size_t lua_objlen(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	switch (ttype(o))
	{
		case LUA_TSTRING:
		{
			return tsvalue(o)->len;
		}
		case LUA_TUSERDATA:
		{
			return uvalue(o)->len;
		}
		case LUA_TTABLE:
		{
			return luaH_getn(hvalue(o));
		}
		case LUA_TNUMBER:
		{
			return luaV_tostring(L, o) ? tsvalue(o)->len : 0;
		}
		default:
		{
			return 0;
		}
	}
}

5.15 lua_tocfunction 获取指定索引的值的C函数的地址

(lapi.c) lua_tocfunction

//typedef int (*lua_CFunction) (lua_State *L);
LUA_API lua_CFunction lua_tocfunction(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	//#define iscfunction(o)	(ttype(o) == LUA_TFUNCTION && clvalue(o)->c.isC)
	//#define clvalue(o)	check_exp(ttisfunction(o), &(o)->value.gc->cl)
	return (!iscfunction(o)) ? NULL : cvalue(o)->c.f;
}

5.16 lua_touserdata 获取指定索引的userdata内容的首地址

(lapi.c) lua_touserdata

LUA_API void* lua_touserdata(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	switch(ttype(o))
	{
		case LUA_TUSERDATA:
		{
			//#define rawuvalue(o)	check_exp(ttisuserdata(o), &(o)->value.gc->u)
			return (rawvalue(o) + 1);//Udata只是一个userdata的头部信息,真正的内容紧随头部之后
		}
		case LUA_TLIGHTUSERDATA:
		{
			//#define pvalue(o)	check_exp(ttislightuserdata(o), (o)->value.p)
			return pvalue(o);
		}
		default:
		{
			return NULL;
		}
	}
}

5.17 lua_tothread 获取指定索引的thread的地址

(lapi.c) lua_tothread

LUA_API lua_State* lua_tothread(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	//#define ttisthread(o)	(ttype(o) == LUA_TTHREAD)
	//#define thvalue(o)	check_exp(ttisthread(o), &(o)->value.gc->th)
	return (!ttisthread(o) ? NULL : thvalue(o));
}

5.18 lua_topointer 获取指定索引的值的有效内容首地址

(lapi.c) lua_topointer

LUA_API const void* lua_topointer(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	switch (ttype(o))
	{
		case LUA_TTABLE:
		{
			return hvalue(o);
		}
		case LUA_TFUNCTION:
		{
			return clvalue(o);
		}
		case LUA_TTHREAD:
		{
			return thvalue(o);
		}
		case LUA_TUSERDATA:
		case LUA_TLIGHTUSERDATA:
		{
			return lua_touserdata(L, idx);
		}
		default:
		{
			return NULL;
		}
	}
}

6 C将值压栈

6.1 lua_pushnil 将nil压栈

(lapi.c) lua_pushnil

LUA_API void lua_pushnil(lua_State* L)
{
	setnilvalue(L->top);//#define setnilvalue(obj) ((obj)->tt=LUA_TNIL)
	api_incr_top(L);
}

6.2 lua_pushnumber 将数字压栈

(lapi.c) lua_pushnumber

LUA_API void lua_pushnumber(lua_State* L, lua_Number n)
{
	//#define setnvalue(obj,x) { TValue *i_o=(obj); i_o->value.n=(x); i_o->tt=LUA_TNUMBER; }
	setnvalue(L->top, n);
	api_incr_top(L);
}

6.3 lua_pushinteger 将整数压栈

(lapi.c) lua_pushinteger

LUA_API void lua_pushinteger(lua_State* L, lua_Integer n)
{
	setnvalue(L->top, cast_num(n));
	api_incr_top(L);
}

6.4 lua_pushlstring 将字符串入栈

(lapi.c) lua_pushlstring

LUA_API void lua_pushlstring(lua_State* L, const char* s, size_t len)
{
	luaC_checkGC(L);
	setsvalue2s(L, L->top, luaS_newlstr(L, s, len));
	api_incr_top(L);
}

6.5 lua_pushstring 将字符串入栈,若为NULL,则将nil压栈

(lapi.c) lua_pushstring

LUA_API void lua_pushstring(lua_State* L, const char* s)
{
	if (s == NULL)
	{
		lua_pushnil(L);
	}
	else
	{
		lua_pushlstring(L, s, strlen(s));
	}
}

6.6 lua_pushvfstring 将格式化字符串压栈

(lapi.c) lua_pushvfstring

LUA_API const char* lua_pushvfstring(lua_State* L, const char* fmt, va_list argp)
{
	luaC_checkGC(L);
	return luaO_pushfstring(L, fmt, argp);
}

(lobject.c) luaO_pushvfstring 根据格式串fmt和va_list向lua栈压入格式化的字符串

const char* luaO_pushvfstring(lua_State* L, const char* fmt, va_list argp)
{
	pushstr(L, "");
	//记录被压栈的字符串数量
	int num_pushed_str = 1;
	for (;;)
	{
		//查找fmt中第一个%的地址,若无,则退出循环
		const char* e = strchr(fmt, '%');
		if (e == NULL)
		{
			break;
		}
		//截取e - fmt长度的字符串压栈
		setsvalue2s(L, L->top, luaS_newlstr(L, fmt, e - fmt));
		incr_top(L);
		//分析百分号的下一个字符
		switch(*(e + 1))
		{
			//若匹配到字符串,字符串压栈
			case 's':
			{
				const char* s = va_arg(argp, char*);
				if (s == NULL)
				{
					s = "(null)";
				}
				pushstr(L, s);
				break;
			}
			//若匹配到字符,字符后接一个\0字符形成一个新的字符串压栈
			case 'c':
			{
				char buff[2];
				buff[0] = cast(char, va_arg(argp, int));
				buff[1] = '\0';
				pushstr(L, buff);
				break;
			}
			//若匹配到整数,则整数压栈
			case 'd':
			{
				setnvalue(L->top, cast_num(va_arg(argp, int)));
				incr_top(L);
				break;
			}
			//若匹配到浮点数,则浮点数压栈
			case 'f':
			{
				setnvalue(L->top, cast_num(va_arg(argp, l_uacNumber)));//l_uacNumber就是double
				incr_top(L);
				break;
			}
			//若匹配到地址值,则将地址值的字符串表示压栈
			case 'p':
			{
				char buff[4*sizoef(void*) + 8];
				sprintf(buff, "%p", va_arg(argp, void*));
				pushstr(L, buff);
				break;
			}
			//匹配到%字符
			case '%':
			{
				pushstr(L, "%");
				break;
			}
			//若%匹配失败,则直接将%字符连同下一个字符,再后接一个\0字符连成字符串,压栈
			default:
			{
				char buff[3];
				buff[0] = '%';
				buff[1] = *(e + 1);
				buff[2] = '\0';
				pushstr(L, buff);
				break;
			}
		}
		num_pushed_str += 2;
		fmt = e + 2;
	}
	pushstr(L, fmt);
	num_pushed_str++;
	//将压入的字符串连接起来
	luaV_concat(L, num_pushed_str, cast_int(L->top - L->base) - 1);
	//虽然压入num_pushed_str个字符串,但是最后要比函数执行前要多1字符串,所以top只是减少num_pushed_str - 1
	L->top -= (num_pushed_str - 1);
	return svalue(L->top - 1);
}

(lobject.c) pushstr 向栈顶压入1个字符串

static void pushstr(lua_State* L, const char* str)
{
	setsvalue2s(L, L->top, luaS_new(L, str));//将栈顶位置设为新创建或者从全字符串表查到的字符串
	incr_top(L);//#define incr_top(L) {luaD_checkstack(L,1); L->top++;}
}

(lvm.c) luaV_concat 连接栈上的字符串

//@param total 需要处理的字符串总数
//@param last 相对于base的偏移
void luaV_concat(lua_State* L, int total, int last)
{
	do
	{
		//计算栈顶位置
		StkId top = L->base + last + 1;
		int n = 2;
		//若top-2不是字符串或数字,或者 top-1不是字符串,则尝试调用TM_CONCAT元方法
		//#define tostring(L,o) ((ttype(o) == LUA_TSTRING) || (luaV_tostring(L, o)))
		if (!(ttisstring(top - 2) || ttisnumber(top - 2)) || !tostring(L, top - 1))
		{
			//如果没有 TM_CONCAT 元方法,则报错
			if (!call_binTM(L, top - 2, top - 1, top - 2, TM_CONCAT))
			{
				luaG_concaterror(L, top - 2, top - 1);
				return;
			}
		}
		//若top-1串长度为0,则不需要连接了,top-2直接转为字符串
		else if (tsvalue(top - 1)->len == 0)
		{
			(void)tostring(L, top - 2);
		}
		else
		{
			//每n个串合并成1个大串(n个串拷贝到缓冲中,n个串只对应1个TString对象)
			//tl为n个串的总长度
			size_t tl = tsvalue(top - 1)->len;
			for (n = 1; n < total && tostring(L, top - n - 1); n++)
			{
				size_t l = tsvalue(top - n - 1)->len;
				if (l >= MAX_SIZET - tl)//#define MAX_SIZET	((size_t)(~(size_t)0)-2)
				{
					luaG_runerror(L, "string length overflow");
				}
				tl += l;
			}
			char* buffer = luaZ_openspace(L, &G(L)->buff, tl);
			tl = 0;
			for (int i = n; i > 0; i--)
			{
				size_t l = tsvalue(top - i)->len;
				memcpy(buffer + tl, svalue(top - i), l);
				tl += l;
			}
			setsvalue2s(L, top - n, luaS_newlstr(L, buffer, tl));
		}
		total -= n - 1;//需要处理的串数量 减少了 n-1个,很好理解,n个字符串合并成了1个(-n+1)
		last -= n - 1;
	} while (total > 1);//合并嘛,肯定是>1才有合并的意义
}

(lvm.c) call_binTM 调用元方法(不要求双方的元方法对等)

static int call_binTM(lua_State* L, const TValue* p1, const TValue* p2, StkId res,  TMS event)
{
	//p1和p2找到一个元方法就行,否则失败,返回0
	const TValue* tm = luaT_gettmbyobj(L, p1, event);
	if (ttisnil(tm))
	{
		tm = luaT_gettmbyobj(L, p2, event);
	}
	if (ttisnil(tm))
	{
		return 0;
	}
	//执行元方法
	callTMres(L, res, tm, p1, p2);
	return 1;
}

(lzio.c) luaZ_openspace 尝试在buff->buffer地址处开辟长度为n字节的缓冲区

char* luaZ_openspace(lua_State* L, Mbuffer* buff, size_t n)
{
	//若需要的字节数 > 缓冲区容量,则扩容
	if (n > buff->buffsize)
	{
		if (n < LUA_MINBUFFER)//#define LUA_MINBUFFER	32
		{
			n = LUA_MINBUFFER;
		}
		//#define luaZ_resizebuffer(L, buff, size) (luaM_reallocvector(L, (buff)->buffer, (buff)->buffsize, size, char),  (buff)->buffsize = size)
		luaZ_resizebuffer(L, buff, n);//为buff->buffer重新分配内存,大小为n
	}
	return buff->buffer;
}

6.7 lua_pushfstring 将格式化字符串压栈

(lapi.c) lua_pushfstring

LUA_API const char* lua_pushfstring(lua_State* L, const char* fmt, ...)
{
	luaC_checkGC(L)
	va_list argp;
	va_start(argp, fmt);
	const char* ret = luaO_pushvfstring(L, fmt, argp);
	va_end(argp);
	return ret;
}

6.8 lua_pushcclosure 将C闭包压栈(从L->top-n到L->top-1的值作为这个C闭包的upvalue,n个upvalue出栈)

(lapi.c) lua_pushcclosure

LUA_API void lua_pushcclosure(lua_State* L, lua_CFunction fn, int n)
{
	luaC_checkGC(L);
	Closure* cl = luaF_newCclosure(L, n, getcurrentv(L));//luaF_newCclosure,getcurrentv见环境章节
	cl->c.f = fn;
	L->top -= n;
	while(n--)
	{
		setobj2n(L, &cl->c.upvalue[n], L->top + n);
	}
	setclvalue(L, L->top, cl);
	api_incr_top(L);
}

6.9 lua_pushboolean 将bool压栈

(lapi.c) lua_pushboolean

LUA_API void lua_pushboolean(lua_State* L, int b)
{
	//#define setbvalue(obj,x) { TValue *i_o=(obj); i_o->value.b=(x); i_o->tt=LUA_TBOOLEAN; }
	setbvalue(L->top, b != 0);
	api_incr_top(L);
}

6.10 lua_pushlightuserdata 将lightuserdata压栈

(lapi.c) lua_pushlightuserdata

LUA_API void lua_pushlightuserdata(lua_State* L, void* p)
{
	setpvalue(L->top, p);
	api_incr_top(L);
}

6.11 lua_pushthread 将当前协程压栈(若当前协程是主协程,则返回true)

(lapi.c) lua_pushthread

LUA_API int lua_pushthread(lua_State* L)
{
	//#define setthvalue(L,obj,x)  { TValue *i_o=(obj); i_o->value.gc=cast(GCObject *, (x)); i_o->tt=LUA_TTHREAD;}
	setthvalue(L, L->top, L);
	api_incr_top(L);
	return (G(L)->mainthread == L);
}

7 通过栈访问Lua

7.1 lua_gettable 以L->top-1处为key,访问idx处的表,将value放在L->top-1处

(lapi.c) lua_gettable

LUA_API void lua_gettable(lua_State* L, int idx)
{
	StkId t = index2adr(L, idx);
	luaV_gettable(L, t, L->top - 1, L->top - 1);
}

(lvm.c) luaV_gettable (含义:val = t[key])

void luaV_gettable(lua_State* L, const TValue* t, TValue* key, StkId val)
{
	//根据key沿着元表一层一层获取value
	for (int loop = 0; loop < MAXTAGLOOP ; loop++)//#define MAXTAGLOOP	100
	{
		const TValue* tm;//用来指向t的__index元方法
		if (ttistable(t))
		{
			Table* h = hvalue(t);
			const TValue* res = luaH_get(h, key);//这里就是所谓的rawget(原始的查表逻辑)//
			//找到的值非空或者有无__index元方法,则直接将找到的值赋给val位置
			if (!ttisnil(res) || (tm = fasttm(L, h->metatble, TM_INDEX) == NULL))
			{
				setobj2s(L, val, res);
				return;
			}
		}
		else if (ttisnil(tm = luaT_gettmbyobj(L, t, TM_INDEX)))
		{
			//若t不是表,但也无__index元方法,则报错
			luaG_typeerror(L, t, "index");//luaG_typeerror见异常章节
		}
		//若__index元方法是函数,则执行这个元方法
		if (ttisfunction(tm))
		{
			callTMres(L, val, tm, t, key);
			return;
		}
		t = tm;
	}
	//若MAXTAGLOOP层都还没找到,则报错
	luaG_runerror(L, "loop in gettable");
}

7.2 lua_getfield 以k为字符串key,访问idx处的表,将value放在L->top处,L->top++

(lapi.c) lua_getfield

LUA_API void lua_getfield(lua_State* L, int idx, const char* k)
{
	StkId t = index2adr(L, idx);
	TValue key;
	setsvalue(L, &key, luaS_new(L, k));
	luaV_gettable(L, t, &key, L->top);
	api_incr_top(L);
}

7.3 lua_rawget 以L->top-1处为key,访问idx处的表,将value放在L->top-1处(不经过元表)

(lapi.c) lua_rawget

LUA_API void lua_rawget(lua_State* L, int idx)
{
	StkId t = index2adr(L, idx);
	setobj2s(L, L->top - 1, luaH_get(hvalue(t), L->top - 1));
}

7.4 lua_rawgeti 以n为数字key,访问idx处的表,将value放在L->top处,L->top++(不经过元表)

(lapi.c) lua_rawgeti

LUA_API void lua_rawgeti(lua_State* L, int idx, int n)
{
	StkId o = index2adr(L, idx);
	setobj2s(L, L->top, luaH_getnum(hvalue(o), n));
	api_incr_top(L);
}

7.5 lua_createtable 创建一个数组长度为narray,哈希长度为nrec的表,压栈

(lapi.c) lua_createtable

LUA_API void lua_createtable(lua_State* L, int narray, int nrec)
{
	luaC_checkGC(L);
	sethvalue(L, L->top, luaH_new(L, narray, nrec));
	api_incr_top(L);
}

7.6 lua_newuserdata 创建一个大小为size的userdata,压栈,返回其内容的首地址

(lapi.c) lua_newuserdata

LUA_API void* lua_newuserdata(lua_State* L, size_t size)
{
	luaC_checkGC(L);
	Udata* u = luaS_newudata(L, size, getcurrenv(L));
	setuvalue(L, L->top, u);
	api_incr_top(L);
	return u + 1;
}

(lstring.c) luaS_newudata 新建一个userdata,并加入主协程的Udata链表

Udata* luaS_newudata(lua_State* L, size_t s, Table* e)
{
	//若太大,则报错
	if (s > MAX_SIZET - sizeof(Udata))//#define MAX_SIZET	((size_t)(~(size_t)0)-2)
	{
		luaM_toobig(L);
	}
	Udata* u = cast(Udata*, luaM_malloc(L, s + sizeof(Udata)));
	u->uv.marked = luaC_white(G(L));//luaC_white见GC章节
	u->uv.tt = LUA_TUSERDATA;
	u->uv.len = s;
	u->uv.metatable = NULL;
	u->uv.env = e;
	//插入到主协程的next链表里(这个链表存所有的udata)
	u->uv.next = G(L)->mainthread->next;
	G(L)->mainthread->next = obj2gco(u);//#define obj2gco(v)	(cast(GCObject *, (v)))
	return u;
}

(lmem.c) luaM_toobig

void* luaM_toobig(lua_State* L)
{
	luaG_runerror(L, "memory allocation error: block too big");
	return NULL;
}

7.7 lua_getmetatable 获取指定索引的值的元表,若成功取得,压栈 (若成功取得,则返回true)

(lapi.c) lua_getmetatable

LUA_API int lua_getmetatable(lua_State* L, int objindex)
{
	Table* mt = NULL;
	int res;
	const TValue* obj = index2adr(L, objindex);
	switch (ttype(obj))
	{
		case LUA_TTABLE:
		{
			mt = hvalue(obj)->metatable;
			break;
		}
		case LUA_TUSERDATA:
		{
			mt = uvalue(obj)->metatable;
			break;
		}
		default:
		{
			mt = G(L)->mt[ttype(obj)];
			break;
		}
	}
	if (mt == NULL)
	{
		res = 0;
	}
	else
	{
		sethvalue(L, L->top, mt);
		api_incr_top(L);
		res = 1;
	}
	return res;
}

7.8 lua_getfenv 获取指定索引值的环境表,若成功取得,压栈;否则将nil压栈

(lapi.c) lua_getfenv

LUA_API void lua_getfenv(lua_State* L, int idx)
{
	StkId o = index2adr(L, idx);
	switch (ttype(o))
	{
		case LUA_TFUNCTION:
		{
			sethvalue(L, L->top, clvalue(o)->c.env);
			break;
		}
		case LUA_TUSERDATA:
		{
			sethvalue(L, L->top, uvalue(o)->env);
			break;
		}
		case LUA_TTHREAD:
		{
			setobj2s(L, L->top, gt(thvalue(o)));
			break;
		}
		default:
		{
			setnilvalue(L->top);
			break;
		}
	}
	api_incr_top(L);
}

8 通过栈操作Lua

8.1 lua_settable 以L->top-2处为key,以L->top-1处为value,操作idx处的表,L->top-=2

(lapi.c) lua_settable

LUA_API void lua_settable(lua_State* L, int idx)
{
	StkId t = index2adr(L, idx);
	luaV_settable(L, t, t->top - 2, L->top - 1);
	L->top -= 2;
}

(lvm.c) luaV_settable (含义:t[key]=val)

void luaV_settable(lua_State* L, const TValue* t, TValue* key, StkId val)
{
	for (int loop = 0; loop < MAXTAGLOOP; loop++)//#define MAXTAGLOOP	100
	{
		const TValue* tm;
		if (ttistable(t))
		{
			Table* h = hvalue(t);
			TValue* oldval = luaH_set(L, h, key);//rawset
			if (!ttisnil(oldval) || (tm = fasttm(L, h->metatble, TM_NEWINDEX)) == NULL)
			{
				setobj2t(L, oldval, val);
				luaC_barriert(L, h, val);//luaC_barriert见GC章节
				return;
			}
		}
		else if (ttisnil(tm = luaT_gettmbyobj(L, t, TM_NEWINDEX)))
		{
			luaG_typeerror(L, t, "index");
		}
		if (ttisfunction(tm))
		{
			callTM(L, tm, t, key, val);
			return;
		}
		t = tm;
	}
	luaG_runerror(L, "loop in settable");
}

(lvm.c) callTM(含义:f(p1,p2,p3))

static void callTM(lua_State* L, const TValue* f, const TValue* p1, const TValue* p2, const TValue* p3)
{
	setobj2s(L, L->top, f);
	setobj2s(L, L->top + 1, p1);
	setobj2s(L, L->top + 2, p2);
	setobj2s(L, L->top + 3, p3);
	luaD_checkstack(L, 4);
	L->top += 4;
	luaD_call(L, L->top - 4, 0);//luaD_call见函数章节
}

8.2 lua_setfield 以k为字符串key,以L->top-1处为value,操作idx处的表,L->top–

(lapi.c) lua_setfield

LUA_API void lua_setfield(lua_State* L, int idx, const char* k)
{
	StkId t = index2adr(L, idx);
	setsvalue(L, &key, luaS_new(L, k));
	luaV_settable(L, t, &key, L->top - 1);
	L->top--;
}

8.3 lua_rawset 以L->top-2处为key,以L->top-1处为value,操作idx处的表(不涉及元表),L->top-=2

(lapi.c) lua_rawset

LUA_API void lua_rawset(lua_State* L, int idx)
{
	StkId t = index2adr(L, idx);
	setobj2t(L, luaH_set(L, hvalue(t), L->top - 2), L->top - 1);
	luaC_barriert(L, hvalue(t), L->top - 1);//luaC_barriert见GC章节
	L->top -= 2;
}

8.4 lua_rawseti 以n为数字键,L->top -1处为value,操作idx处的表(不涉及元表),L->top–

(lapi.c) lua_rawseti

LUA_API void lua_rawseti(lua_State* L, int idx, int n)
{
	StkId o = index2adr(L, idx);
	setobj2t(L, luaH_setnum(L, hvalue(o), n), L->top - 1);
	luaC_barriert(L, hvalue(o), L->top - 1);
	L->top--;
}

8.5 lua_setmetatable 将L->top-1处的值,设置为objindex处的值的元表,L->top–

(lapi.c) lua_setmetatable

LUA_API int lua_setmetatable(lua_State* L, int objindex)
{
	Table* mt;
	TValue* obj = index2adr(L, idx);
	if (ttisnil(L->top - 1))
	{
		mt = NULL;
	}
	else
	{
		mt = hvalue(L->top - 1);
	}
	switch (ttype(obj))
	{
		case LUA_TTABLE:
		{
			hvalue(obj)->metatable = mt;
			if (mt)
			{
				luaC_objbarriert(L, hvalue(obj), mt);
			}
			break;
		}
		case LUA_TUSERDATA:
		{
			uvalue(obj)->metatable = mt;
			if (mt)
			{
				luaC_objbarriert(L, rawuvalue(obj), mt);
			}
			break;
		}
		default:
		{
			G(L)->mt[ttype(o)] = mt;
			break;
		}
	}
	L->top--;
	return 1;
}

8.6 lua_setfenv 将L->top-1处的值,设置为objindex处的值的环境表,L->top–

(lapi.c) lua_setfenv

LUA_API int lua_setfenv(lua_State* L, int idx)
{
	int res = 1;
	StkId o = index2adr(L, idx);
	switch (ttype(o))
	{
		case LUA_TFUNCTION:
		{
			clvalue(o)->c.env = hvalue(L->top - 1);
			break;
		}
		case LUA_TUSERDATA:
		{
			uvalue(o)->env = hvalue(L->top - 1);
			break;
		}
		case LUA_TTHREAD:
		{
			sethvalue(L, gt(thvalue(o)), hvalue(L->top - 1));
			break;
		}
		default:
		{
			res = 0;
			break;
		}
	}
	if (res)
	{
		luaC_objbarriert(L, gcvalue(o), hvalue(L->top - 1));
	}
	L->top--;
	return res;
}
  • 7
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值