GSTMT386.C
上传用户:hlzzc88
上传日期:2007-01-06
资源大小:220k
文件大小:20k
源码类别:

编译器/解释器

开发平台:

Others

  1. /*
  2.  * 68K/386 32-bit C compiler.
  3.  *
  4.  * copyright (c) 1997, David Lindauer
  5.  * 
  6.  * This compiler is intended for educational use.  It may not be used
  7.  * for profit without the express written consent of the author.
  8.  *
  9.  * It may be freely redistributed, as long as this notice remains intact
  10.  * and either the original sources or derived sources 
  11.  * are distributed along with any executables derived from the originals.
  12.  *
  13.  * The author is not responsible for any damages that may arise from use
  14.  * of this software, either idirect or consequential.
  15.  *
  16.  * v1.35 March 1997
  17.  * David Lindauer, gclind01@starbase.spd.louisville.edu
  18.  *
  19.  * Credits to Mathew Brandt for original K&R C compiler
  20.  *
  21.  */
  22. #include        <stdio.h>
  23. #include        "expr.h"
  24. #include        "c.h"
  25. #include        "gen386.h"
  26. #include  "diag.h"
  27. extern TABLE gsyms;
  28. extern int stdaddrsize;
  29. extern int global_flag;
  30. extern int fsave_mask, save_mask;
  31. extern TYP              stdfunc;
  32. extern struct amode     push[], pop[];
  33. extern OCODE *peep_tail, *peep_head, *peep_insert;
  34. extern long stackdepth,framedepth;
  35. extern SYM *currentfunc;
  36. extern int regs[3];
  37. extern int prm_cplusplus,prm_linkreg;
  38. extern int firstlabel, nextlabel;
  39. extern long lc_maxauto;
  40. int funcfloat;
  41. static int     breaklab;
  42. static int     contlab;
  43. static int     retlab;
  44. static int diddef;
  45. static long gswitchbottom, gswitchcount;
  46. static long gswitchtop;
  47. AMODE    *makedreg(int r)
  48. /*
  49.  *      make an address reference to a data register.
  50.  */
  51. {       AMODE    *ap;
  52.         ap = xalloc(sizeof(AMODE));
  53.         ap->mode = am_dreg;
  54.         ap->preg = r;
  55.         return ap;
  56. }
  57. AMODE    *makefreg(int r)
  58. /*
  59.  *      make an address reference to a data register.
  60.  */
  61. {       AMODE    *ap;
  62.         ap = xalloc(sizeof(AMODE));
  63.         ap->mode = am_freg;
  64.         ap->preg = r;
  65.         return ap;
  66. }
  67. AMODE    *make_direct(int i)
  68. /*
  69.  *      make a direct reference to an immediate value.
  70.  */
  71. {       return make_offset(makenode(en_icon,(char *)i,0));
  72. }
  73. AMODE    *make_strlab(char *s)
  74. /*
  75.  *      generate a direct reference to a string label.
  76.  */
  77. {       AMODE    *ap;
  78.         ap = xalloc(sizeof(AMODE));
  79.         ap->mode = am_direct;
  80.         ap->offset = makenode(en_nacon,s,0);
  81.         return ap;
  82. }
  83. void genwhile(SNODE *stmt)
  84. /*
  85.  *      generate code to evaluate a while statement.
  86.  */
  87. {       int     lab1, lab2, lab3;
  88.         initstack();            /* initialize temp registers */
  89.         lab1 = contlab;         /* save old continue label */
  90.         contlab = nextlabel++;  /* new continue label */
  91.         if( stmt->s1 != 0 )      /* has block */
  92.                 {
  93.          lab2 = breaklab;        /* save old break label */
  94.                 breaklab = nextlabel++;
  95. gen_code(op_jmp,0,make_label(contlab),0);
  96. lab3 = nextlabel++;
  97. gen_label(lab3);
  98.                 genstmt(stmt->s1);
  99. gen_label(contlab);
  100. if (stmt->lst)
  101.   gen_line(stmt->lst);
  102.                 initstack();
  103.                 truejp(stmt->exp,lab3);
  104.                 gen_label(breaklab);
  105.                 breaklab = lab2;        /* restore old break label */
  106.                 }
  107.         else         /* no loop code */
  108.                 {
  109. if (stmt->lst)
  110.   gen_line(stmt->lst);
  111.          gen_label(contlab);
  112.                 initstack();
  113.                 truejp(stmt->exp,contlab);
  114.                 }
  115.         contlab = lab1;         /* restore old continue label */
  116. }
  117. void gen_for(SNODE *stmt)
  118. /*
  119.  *      generate code to evaluate a for loop
  120.  */
  121. {       int     old_break, old_cont, exit_label, loop_label;
  122.         old_break = breaklab;
  123.         old_cont = contlab;
  124.         loop_label = nextlabel++;
  125.         exit_label = nextlabel++;
  126.         contlab = nextlabel++;
  127.         initstack();
  128.         if( stmt->label != 0 )
  129.                 gen_expr(stmt->label,F_ALL | F_NOVALUE
  130.                         ,natural_size(stmt->label));
  131.         gen_code(op_jmp,0,make_label(contlab),0);
  132.         gen_label(loop_label);
  133.         if( stmt->s1 != 0 ) {
  134. breaklab = exit_label;
  135.                 genstmt(stmt->s1);
  136. }
  137.         initstack();
  138.         if( stmt->s2 != 0 )
  139.                 gen_expr(stmt->s2,F_ALL | F_NOVALUE,natural_size(stmt->s2));
  140. gen_label(contlab);
  141. if (stmt->lst)
  142.   gen_line(stmt->lst);
  143.         initstack();
  144.         if( stmt->exp != 0 )
  145.                 truejp(stmt->exp,loop_label);
  146. else
  147. gen_code(op_jmp,0,make_label(loop_label),0);
  148. gen_label(exit_label);
  149.   breaklab = old_break;
  150. contlab = old_cont;
  151. }
  152. void genif(SNODE *stmt)
  153. /*
  154.  *      generate code to evaluate an if statement.
  155.  */
  156. {       int     lab1, lab2;
  157.         lab1 = nextlabel++;     /* else label */
  158.         lab2 = nextlabel++;     /* exit label */
  159.         initstack();            /* clear temps */
  160.         falsejp(stmt->exp,lab1);
  161.         genstmt(stmt->s1);
  162.         if( stmt->s2 != 0 )             /* else part exists */
  163.                 {
  164.                 gen_code(op_jmp,0,make_label(lab2),0);
  165.                 gen_label(lab1);
  166.                 genstmt(stmt->s2);
  167.                 gen_label(lab2);
  168.                 }
  169.         else                            /* no else code */
  170.                 gen_label(lab1);
  171. }
  172. void gendo(SNODE *stmt)
  173. /*
  174.  *      generate code for a do - while loop.
  175.  */
  176. {       int     oldcont, oldbreak;
  177.         oldcont = contlab;
  178.         oldbreak = breaklab;
  179.         contlab = nextlabel++;
  180.         gen_label(contlab);
  181.         if( stmt->s1 != 0 && stmt->s1->next != 0 )
  182.                 {
  183.                 breaklab = nextlabel++;
  184.                 genstmt(stmt->s1);      /* generate body */
  185.                 initstack();
  186.                 truejp(stmt->exp,contlab);
  187.                 gen_label(breaklab);
  188.                 }
  189.         else
  190.                 {
  191.                 genstmt(stmt->s1);
  192.                 initstack();
  193.                 truejp(stmt->exp,contlab);
  194.                 }
  195.         breaklab = oldbreak;
  196.         contlab = oldcont;
  197. }
  198. AMODE *set_symbol(char *name,int flag)
  199. {
  200. SYM *sp;
  201. AMODE *ap;
  202.         sp = gsearch(name);
  203.         if( sp == 0 )
  204.                 {
  205.                 ++global_flag;
  206.                 sp = xalloc(sizeof(SYM));
  207.                 sp->tp = &stdfunc;
  208.                 sp->name = name;
  209. if (flag)
  210. sp->storage_class = sc_externalfunc;
  211. else
  212.                  sp->storage_class = sc_external;
  213. sp->extflag = 1;
  214.                 insert(sp,&gsyms);
  215.                 --global_flag;
  216.                 }
  217. ap = make_strlab(name);
  218. ap->mode = am_immed;
  219. return ap;
  220. }
  221. void call_library(char *lib_name)
  222. /*
  223.  *      generate a call to a library routine.
  224.  */
  225. {
  226. AMODE *ap;
  227. ap = set_symbol(lib_name,1);
  228.         gen_code(op_call,0,ap,0);
  229. }
  230. int analyzeswitch(SNODE *stmt)
  231. {
  232. gswitchbottom = 0x7fffffff;
  233. gswitchtop = 0x80000000;
  234. gswitchcount = 0;
  235. stmt = stmt->s1;
  236. while (stmt) {
  237. if (!stmt->s2) {
  238. gswitchcount++;
  239. if ((long)stmt->label < gswitchbottom)
  240. gswitchbottom = (int)stmt->label;
  241. if ((long)stmt->label > gswitchtop)
  242. gswitchtop = (int)stmt->label;
  243. }
  244. stmt = stmt->next;
  245. }
  246. gswitchtop++;
  247. if (gswitchcount == 0)
  248. return(0);
  249. if (gswitchcount > 3)
  250. if (gswitchcount*10/(gswitchtop-gswitchbottom) >= 8)
  251. return(1);
  252. return(2);
  253. }
  254. void bingen(int lower, int avg, int higher,AMODE *ap1, int deflab, int size,long * switchids,int * switchlabels,int *switchbinlabels)
  255. {
  256. AMODE *ap2 = make_immed(switchids[avg]);
  257. AMODE *ap3 = make_label(switchlabels[avg]);
  258. if (switchbinlabels[avg] != -1)
  259. gen_label(switchbinlabels[avg]);
  260. gen_code(op_cmp,4,ap1,ap2);
  261. gen_code(op_je,0,ap3,0);
  262. if (avg == lower) {
  263. ap3 = make_label(deflab);
  264. gen_code(op_jmp,0,ap3,0);
  265. }
  266. else {
  267. int avg1 = (lower + avg)/2;
  268. int avg2 = (higher + avg+1)/2;
  269. if (avg+1 < higher) 
  270. ap3 = make_label(switchbinlabels[avg2]=nextlabel++);
  271. else
  272. ap3 = make_label(deflab);
  273. if (size < 0)
  274. gen_code(op_jg,0,ap3,0);
  275. else
  276. gen_code(op_ja,0,ap3,0);
  277. bingen(lower,avg1,avg,ap1,deflab,size,switchids,switchlabels,switchbinlabels);
  278. if (avg+1 < higher)
  279. bingen(avg+1,avg2,higher,ap1,deflab,size,switchids,switchlabels,switchbinlabels);
  280. }
  281. }
  282. void genbinaryswitch(SNODE *stmt, int deflab)
  283. {
  284. int curlab, i=0,j,size = natural_size(stmt->exp);
  285. AMODE *ap1;
  286.   long switchbottom=gswitchbottom, switchcount=gswitchcount;
  287.   long switchtop=gswitchtop;
  288.   int *switchlabels=0;
  289.   long *switchids=0;
  290.   int *switchbinlabels=0;
  291. curlab = nextlabel++;
  292.   initstack();
  293.   ap1 = gen_expr(stmt->exp,F_DREG,4);
  294. global_flag++;
  295. switchlabels = xalloc((switchcount) * sizeof(int));
  296. switchbinlabels = xalloc((switchcount) * sizeof(int));
  297. switchids = xalloc((switchcount) * sizeof(long));
  298. global_flag--;
  299. stmt = stmt->s1;
  300. while (stmt) {
  301.                 if( stmt->s2 )          /* default case ? */
  302.                         {
  303.                         stmt->label = (SNODE *) deflab;
  304. diddef = TRUE;
  305.                         }
  306.                 else
  307.                         {
  308. switchlabels[i] = curlab;
  309. switchbinlabels[i] = -1;
  310. switchids[i++] = (int)stmt->label;
  311. stmt->label = (SNODE *)curlab;
  312.                         }
  313.                 if( stmt->next != 0 )
  314.                         curlab = nextlabel++;
  315.                 stmt = stmt->next;
  316.                 }
  317. for (i=0; i < switchcount; i++)
  318. for (j=i+1; j < switchcount; j++)
  319. if (switchids[j] < switchids[i]) {
  320. int temp = switchids[i];
  321. switchids[i] = switchids[j];
  322. switchids[j] = temp;
  323. temp = switchlabels[i];
  324. switchlabels[i] = switchlabels[j];
  325. switchlabels[j] = temp;
  326. }
  327. bingen(0,(switchcount)/2,switchcount,ap1,deflab,size,switchids,switchlabels,switchbinlabels);
  328. freeop(ap1);
  329. }
  330. void gencompactswitch(SNODE *stmt, int deflab)
  331. {       int             tablab,curlab,i, size = natural_size(stmt->exp);
  332.         AMODE    *ap,*ap2;
  333.   long switchbottom=gswitchbottom, switchcount=gswitchcount;
  334.   long switchtop=gswitchtop;
  335.   int *switchlabels=0;
  336. tablab = nextlabel++;
  337.         curlab = nextlabel++;
  338.         initstack();
  339.         ap = gen_expr(stmt->exp,F_DREG | F_VOL,4);
  340. initstack();
  341. if (switchbottom)
  342. gen_code(op_sub,4,ap,make_immed(switchbottom));
  343. if (size < 0)
  344. gen_code(op_jl,0,make_label(deflab),0);
  345. else
  346. gen_code(op_jb,0,make_label(deflab),0);
  347. gen_code(op_cmp,4,ap,make_immed(switchtop-switchbottom));
  348. if (size < 0)
  349. gen_code(op_jge,0,make_label(deflab),0);
  350. else
  351. gen_code(op_jnc,0,make_label(deflab),0);
  352. gen_code(op_shl,4,ap,make_immed(2));
  353. ap2 = xalloc(sizeof(AMODE));
  354. ap->mode = am_indisp;
  355. ap2->preg = ap->preg;
  356. ap->offset = makenode(en_labcon,(char *)tablab,0);
  357. gen_code(op_jmp,4,ap,0);
  358. initstack();
  359. align(4);
  360. gen_label(tablab);
  361. switchlabels = xalloc((switchtop-switchbottom) * sizeof(int));
  362. for (i=switchbottom; i < switchtop; i++) {
  363. switchlabels[i-switchbottom] = deflab;
  364. }
  365. stmt = stmt->s1;
  366. while (stmt) {
  367.                 if( stmt->s2 )          /* default case ? */
  368.                         {
  369.                         stmt->label = (SNODE *)deflab;
  370. diddef = TRUE;
  371.                         }
  372.                 else
  373.                         {
  374. switchlabels[(int)stmt->label-switchbottom] = curlab;
  375. stmt->label = (SNODE *)curlab;
  376.                         }
  377.                 if(stmt->next != 0 )
  378.                         curlab = nextlabel++;
  379.                 stmt = stmt->next;
  380.                 }
  381. for (i=0; i < switchtop-switchbottom; i++)
  382. gen_code(op_dd,4,make_label(switchlabels[i]),0);
  383. }
  384. void gencase(SNODE *stmt)
  385. /*
  386.  *      generate all cases for a switch statement.
  387.  */
  388. {       while( stmt != 0 )
  389.                 {
  390.                 gen_label((int)stmt->label);
  391.                 if( stmt->s1 != 0 )
  392.                         {
  393.                         genstmt(stmt->s1);
  394.                         }
  395.                 stmt = stmt->next;
  396.                 }
  397. }
  398. void genxswitch(SNODE *stmt)
  399. /*
  400.  *      analyze and generate best switch statement.
  401.  */
  402. {       int     oldbreak;
  403. int olddiddef = diddef;
  404. int deflab = nextlabel++;
  405.         oldbreak = breaklab;
  406.         breaklab = nextlabel++;
  407. diddef = FALSE;
  408. switch (analyzeswitch(stmt)) {
  409.    case 2:
  410. genbinaryswitch(stmt,deflab);
  411. break;
  412. case 1:
  413. gencompactswitch(stmt,deflab);
  414. break;
  415. case 0:
  416. if (stmt->s1)
  417. stmt->s1->label = (SNODE *) nextlabel++;
  418. break;
  419. }
  420.         gencase(stmt->s1);
  421. if (!diddef)
  422. gen_label(deflab);
  423.         gen_label(breaklab);
  424.         breaklab = oldbreak;
  425. diddef = olddiddef;
  426. }
  427. void genreturn(SNODE *stmt, int flag)
  428. /*
  429.  *      generate a return statement.
  430.  */
  431. {       AMODE    *ap,*ap1;
  432. int size;
  433.         if( stmt != 0 && stmt->exp != 0 ) {
  434.                 initstack();
  435. if (currentfunc->tp->btp && currentfunc->tp->btp->type != bt_void && (currentfunc->tp->btp->type == bt_struct || currentfunc->tp->btp->type == bt_union)) {
  436. size = currentfunc->tp->btp->size;
  437. ap = gen_expr(stmt->exp,F_ALL,4);
  438. if (!(save_mask & 0x40))
  439. gen_push(ESI,am_dreg,0);                                  
  440. if (!(save_mask & 0x80))
  441. gen_push(EDI,am_dreg,0);                                  
  442. if (prm_linkreg) {
  443. ap1 = xalloc(sizeof(AMODE));
  444. ap1->preg = EBP;
  445. ap1->mode = am_indisp;
  446. if (currentfunc->pascaldefn && currentfunc->tp->lst.head && currentfunc->tp->lst.head != (SYM *) -1) 
  447. ap1->offset = makenode(en_icon,(char *)(currentfunc->tp->lst.head->value.i+((currentfunc->tp->lst.head->tp->size+3) &0xFFFFFFFCL)),0);
  448. else
  449. ap1->offset = makenode(en_icon,(char *)8,0);
  450. }
  451. else
  452. if (currentfunc->pascaldefn && currentfunc->tp->lst.head && currentfunc->tp->lst.head != (SYM *) -1) 
  453. ap1 = make_stack(-stackdepth-framedepth-currentfunc->tp->lst.head->value.i-((currentfunc->tp->lst.head->tp->size+3) & 0xfffffffcL));
  454. else
  455. ap1 = make_stack(-stackdepth-framedepth);
  456. gen_code(op_mov,4,makedreg(ESI),ap);                   
  457. gen_code(op_mov,4,makedreg(EDI),ap1);
  458. gen_code(op_mov,4,makedreg(EAX),makedreg(EDI));
  459. gen_code(op_mov,4,makedreg(ECX),make_immed(size));  
  460. gen_code(op_cld,0,0,0);
  461. gen_code(op_rep,1,0,0);                                   
  462. gen_code(op_movsb,1,0,0);                               
  463. if (!(save_mask & 0x80))
  464. gen_pop(EDI,am_dreg,0);
  465. if (!(save_mask & 0x40))
  466. gen_pop(ESI,am_dreg,0);
  467. }
  468. else {
  469. size = currentfunc->tp->btp->size;
  470.                  ap = gen_expr(stmt->exp,F_DREG | F_FREG,size);
  471. if (size > 4) {
  472. if (ap->mode != am_freg)
  473. gen_code(op_fld,size,ap,0);
  474. }
  475.                  else
  476.                   if( ap->mode != am_dreg || ap->preg != 0 )
  477.                          gen_code(op_mov,size,makedreg(0),ap);
  478.                 }
  479. freeop(ap);
  480. }
  481. if (flag) {
  482.         if( retlab != -1 )
  483.                 gen_label(retlab);
  484.           if( fsave_mask != 0 )
  485. diag("Float restore in return");
  486. if (!prm_linkreg && lc_maxauto)
  487. gen_code(op_add,4,makedreg(ESP),make_immed(lc_maxauto));
  488. if (currentfunc->intflag) {
  489. gen_code(op_popad,0,0,0);
  490. if (prm_linkreg && (lc_maxauto || currentfunc->tp->lst.head && currentfunc->tp->lst.head != (SYM *)-1)) {
  491. gen_code(op_leave,0,0,0);
  492. }
  493.             gen_code(op_iretd,0,0,0);
  494. }
  495. else {
  496.             if( save_mask != 0 )
  497. popregs(save_mask);
  498. if (prm_linkreg && (lc_maxauto || currentfunc->tp->lst.head && currentfunc->tp->lst.head != (SYM *)-1)) {
  499. gen_code(op_leave,0,0,0);
  500. }
  501. if (currentfunc->pascaldefn) {
  502. long retsize = 0;
  503. if (currentfunc->tp->lst.head && currentfunc->tp->lst.head != (SYM *)-1) {
  504. retsize = currentfunc->tp->lst.head->value.i+((currentfunc->tp->lst.head->tp->size + 3) & 0xfffffffcL);
  505. if (prm_linkreg)
  506. retsize -= 8;
  507. }
  508. if (currentfunc->tp->btp && currentfunc->tp->btp->type != bt_void && (currentfunc->tp->btp->type == bt_struct || currentfunc->tp->btp->type == bt_union)) 
  509. retsize += 4;
  510. if (retsize) {
  511.              gen_code(op_ret,0,make_immed(retsize),0);
  512. return;
  513. }
  514. }
  515.             gen_code(op_ret,0,0,0);
  516. }
  517. }
  518.         else {
  519. if (retlab == -1)
  520.             retlab = nextlabel++;
  521.           gen_code(op_jmp,0,make_label(retlab),0);
  522. }
  523. }
  524. void genstmt(SNODE *stmt)
  525. /*
  526.  *      genstmt will generate a statement and follow the next pointer
  527.  *      until the block is generated.
  528.  */
  529. {
  530.    while( stmt != 0 )
  531.                 {
  532.                 switch( stmt->stype )
  533.                         {
  534. case st_block:
  535. genstmt(stmt->exp);
  536. break;
  537.                         case st_label:
  538.                                 gen_label((int)stmt->label);
  539.                                 break;
  540.                         case st_goto:
  541.                                 gen_code(op_jmp,0,make_label((int)stmt->label),0);
  542.                                 break;
  543.                         case st_expr:
  544.                                 initstack();
  545.                                 gen_expr(stmt->exp,F_ALL | F_NOVALUE,
  546.                                         natural_size(stmt->exp));
  547.                                 break;
  548.                         case st_return:
  549.                                 genreturn(stmt,0);
  550.                                 break;
  551.                         case st_if:
  552.                                 genif(stmt);
  553.                                 break;
  554.                         case st_while:
  555.                                 genwhile(stmt);
  556.                                 break;
  557.                         case st_do:
  558.                                 gendo(stmt);
  559.                                 break;
  560.                         case st_for:
  561.                                 gen_for(stmt);
  562.                                 break;
  563.                         case st_continue:
  564.                                 gen_code(op_jmp,0,make_label(contlab),0);
  565.                                 break;
  566.                         case st_break:
  567.                                 gen_code(op_jmp,0,make_label(breaklab),0);
  568.                                 break;
  569.                         case st_switch:
  570.                                 genxswitch(stmt);
  571.                                 break;
  572. case st_line:
  573.   gen_line(stmt);
  574. break;
  575. case st_asm:
  576. if (stmt->exp)
  577. add_peep(stmt->exp);
  578. break;
  579. case st__genword:
  580. gen_code(op_genword,0,make_immed((int)stmt->exp),0);
  581. break;
  582.                         default:
  583.                                 diag("unknown statement.");
  584.                                 break;
  585.                         }
  586.                 stmt = stmt->next;
  587.                 }
  588. }
  589. #ifdef CPLUSPLUS
  590. void scppinit(void)
  591. {
  592. if (!strcmp(currentfunc->name,"_main")) {
  593. AMODE *ap1,*ap2,*ap3,*ap4;
  594. int lbl = nextlabel++;
  595. initstack();
  596. ap1 = temp_data();
  597. ap4 = xalloc(sizeof(AMODE));
  598. ap4->preg = ap1->preg;
  599. ap4->mode = am_indisp;
  600. ap4->offset = makenode(en_icon,0,0);
  601. ap2 = set_symbol("cppistart",0);
  602. ap3 = set_symbol("cppiend",0);
  603. gen_code(op_mov,4,ap1,ap2);
  604. gen_label(lbl);
  605. gen_code(op_push,4,ap1,0);
  606. gen_code(op_call,4,ap4,0);
  607. gen_code(op_pop,4,ap1,0);
  608. gen_code(op_add,4,ap1,make_immed(4));
  609. gen_code(op_cmp,4,ap1,ap3);
  610. gen_code(op_jb,0,make_label(lbl),0);
  611. freeop(ap1);
  612. }
  613. }
  614. #endif
  615. void genfunc(SNODE *stmt)
  616. /*
  617.  *      generate a function body.
  618.  */
  619. {       retlab = contlab = breaklab = -1;
  620. funcfloat = 0;
  621. init_muldivval();
  622. stackdepth = 0;
  623. if (stmt->stype == st_line) {
  624. gen_line(stmt);
  625. stmt = stmt->next;
  626. }
  627. gen_codelab(currentfunc);  /* name of function */
  628.         opt1(stmt); /* push args & also subtracts SP */
  629. #ifdef CPLUSPLUS
  630. if (prm_cplusplus)
  631. scppinit();
  632. #endif
  633.         genstmt(stmt);
  634.         genreturn(0,1);
  635. }