转赋值表达式解析的流程
轉(zhuǎn)自:http://www.cnblogs.com/nazhizq/p/6520072.html
上節(jié)說到表達(dá)式的解析問題,exprstate函數(shù)用于解析普通的賦值表達(dá)式。lua語言支持多變量賦值。本文先從單變量賦值表達(dá)式講起。
a = 1 b = 2 c = a + b對于簡單的兩個數(shù)的求和過程,lua源碼是如何解析的呢?
首先,當(dāng)詞法分析獲取到第一個token為‘a(chǎn)’的類型是TK_NAME(285),然后是chunk函數(shù),statment函數(shù),走到exprstate函數(shù):
static void exprstat (LexState *ls) { /* stat -> func | assignment */FuncState *fs = ls->fs; struct LHS_assign v;/*保存等號左邊的變量名*/primaryexp(ls, &v.v);/*處理等號左邊的變量名*/if (v.v.k == VCALL) /* stat -> func */SETARG_C(getcode(fs, &v.v), 1); /* call statement uses no results */else { /* stat -> assignment */v.prev = NULL; assignment(ls, &v, 1); } }其中,LHS_assign是一個包含expdesc結(jié)構(gòu)體的鏈表,擁有指向另一個變量的指針*prev。每個expdesc代表一個變量,該鏈表用于保存等式左邊的所有變量。
表達(dá)式分割的函數(shù)最終會從primaryexp進(jìn)入到prefixexp函數(shù)里,由于當(dāng)前的token值為TK_NAME=285,走到singlevar,即表示單變量的解析函數(shù)。
static void singlevar (LexState *ls, expdesc *var) {TString *varname = str_checkname(ls);FuncState *fs = ls->fs;if (singlevaraux(fs, varname, var, 1) == VGLOBAL)var->u.s.info = luaK_stringK(fs, varname); /* info points to global name *//*指向變量名在寄存器的索引值*/ }luaK_stringK的最終返回值為變量名'a'在fs->f->k這個數(shù)組中的索引值,保存在var->u.s.info。這個值在生成字節(jié)碼時會用到。
然后是singlevaraux,第一次進(jìn)入改函數(shù),fs != NULL,進(jìn)入else,在當(dāng)前層次查找變量,找不到自動遞歸到上層,即fs->prev指向的上層fs,最后返回VGLOBAL。
static int singlevaraux (FuncState *fs, TString *n, expdesc *var, int base) {if (fs == NULL) { /* no more levels? */init_exp(var, VGLOBAL, NO_REG); /* default is global variable */return VGLOBAL;}else {int v = searchvar(fs, n); /* look up at current level */if (v >= 0) {init_exp(var, VLOCAL, v);if (!base)markupval(fs, v); /* local will be used as an upval */return VLOCAL;}else { /* not found at current level; try upper one */if (singlevaraux(fs->prev, n, var, 0) == VGLOBAL)/**/return VGLOBAL;var->u.s.info = indexupvalue(fs, n, var); /* else was LOCAL or UPVAL */var->k = VUPVAL; /* upvalue in this level */return VUPVAL;}} }最后通過luaK_stringK函數(shù)調(diào)用addK函數(shù)對變量‘a(chǎn)’進(jìn)行處理。?luaH_set()一開始調(diào)用luaH_get()在全局變量表中查找該value是否存在, 存在則直接返回值.不存在則調(diào)用newkey()完成添加動作。最終變量名'a'會放到f->k這個數(shù)組中,并且會返回對應(yīng)的索引,然后講索引保存到字節(jié)碼中。
static int addk (FuncState *fs, TValue *k, TValue *v) {lua_State *L = fs->L;TValue *idx = luaH_set(L, fs->h, k);Proto *f = fs->f;int oldsize = f->sizek;if (ttisnumber(idx)) {lua_assert(luaO_rawequalObj(&fs->f->k[cast_int(nvalue(idx))], v));return cast_int(nvalue(idx));}else { /* constant not found; create a new entry */setnvalue(idx, cast_num(fs->nk));luaM_growvector(L, f->k, fs->nk, f->sizek, TValue,MAXARG_Bx, "constant table overflow");while (oldsize < f->sizek) setnilvalue(&f->k[oldsize++]);setobj(L, &f->k[fs->nk], v);luaC_barrier(L, f, v);return fs->nk++;} }這時候,回到exprstat函數(shù),等號左邊的變量名處理完了。然后處理等號右邊的值,調(diào)用assignment函數(shù)賦值。如果下一個token是逗號,說明是多變量賦值。本例中是單變量。nexps = explist1(ls, &e);用于處理等號右邊的值的表達(dá)式,將結(jié)果存入&e中,并返回右值的個數(shù),然后判斷是表達(dá)式的個數(shù)是否和右值的個數(shù)相等。
static void assignment (LexState *ls, struct LHS_assign *lh, int nvars) {expdesc e;check_condition(ls, VLOCAL <= lh->v.k && lh->v.k <= VINDEXED,"syntax error");if (testnext(ls, ',')) { /* assignment -> `,' primaryexp assignment */struct LHS_assign nv;nv.prev = lh;primaryexp(ls, &nv.v);if (nv.v.k == VLOCAL)check_conflict(ls, lh, &nv.v);assignment(ls, &nv, nvars+1);}else { /* assignment -> `=' explist1 */int nexps;checknext(ls, '=');nexps = explist1(ls, &e);/*解析等號右邊的值*/if (nexps != nvars) {adjust_assign(ls, nvars, nexps, &e);if (nexps > nvars)ls->fs->freereg -= nexps - nvars; /* remove extra values */}else {luaK_setoneret(ls->fs, &e); /* close last expression */luaK_storevar(ls->fs, &lh->v, &e);/*生成指令*/return; /* avoid default */}}init_exp(&e, VNONRELOC, ls->fs->freereg-1); /* default assignment */luaK_storevar(ls->fs, &lh->v, &e); }表達(dá)式分析函數(shù)是通過subexpr函數(shù)進(jìn)行遞歸下降分析。這個知識點(diǎn)以后會專門來講,現(xiàn)在由于只是簡單賦值,不會涉及到運(yùn)算符優(yōu)先級的問題。本例中最終調(diào)用的是simpleexp函數(shù),進(jìn)入case?TK_NUMBER:
static void simpleexp (LexState *ls, expdesc *v) {/* simpleexp -> NUMBER | STRING | NIL | true | false | ... |constructor | FUNCTION body | primaryexp */switch (ls->t.token) {case TK_NUMBER: {init_exp(v, VKNUM, 0);/*傳入寄存器位置為0*/v->u.nval = ls->t.seminfo.r;/*將浮點(diǎn)數(shù)1.0賦值給v->u.navl*/break;}case …………………………}luaX_next(ls); }最后,luaK_storevar函數(shù)會將右值保存在寄存器,并生成相應(yīng)的指令碼
void luaK_storevar (FuncState *fs, expdesc *var, expdesc *ex) {switch (var->k) {case VLOCAL: {freeexp(fs, ex);exp2reg(fs, ex, var->u.s.info);return;}case VUPVAL: {int e = luaK_exp2anyreg(fs, ex);luaK_codeABC(fs, OP_SETUPVAL, e, var->u.s.info, 0);break;}case VGLOBAL: {/*本例中是全局變量*/int e = luaK_exp2anyreg(fs, ex);//返回寄存器索引luaK_codeABx(fs, OP_SETGLOBAL, e, var->u.s.info);//生成指令break;}case VINDEXED: {int e = luaK_exp2RK(fs, ex);luaK_codeABC(fs, OP_SETTABLE, var->u.s.info, var->u.s.aux, e);break;}default: {lua_assert(0); /* invalid var kind to store */break;}}freeexp(fs, ex); }最后調(diào)用luaK_codeABx生成指令,關(guān)于指令問題,下回再敘。
int luaK_codeABx (FuncState *fs, OpCode o, int a, unsigned int bc) {lua_assert(getOpMode(o) == iABx || getOpMode(o) == iAsBx);lua_assert(getCMode(o) == OpArgN);return luaK_code(fs, CREATE_ABx(o, a, bc), fs->ls->lastline); }?
《新程序員》:云原生和全面數(shù)字化實(shí)踐50位技術(shù)專家共同創(chuàng)作,文字、視頻、音頻交互閱讀總結(jié)
以上是生活随笔為你收集整理的转赋值表达式解析的流程的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: malloc基本实现
- 下一篇: Laravel Scheduling P