SmaliParser.js

  1. const _fs_ = require('fs');
  2. const _es_ = require('event-stream');
  3. var ut = require("./Utils.js");
  4. var OPCODE = require("./Opcode.js");
  5. var CONST = require("./CoreConst.js");
  6. const CLASS = require("./CoreClass.js");
  7. const Event = require("./Event.js");
  8. const Logger = require("./Logger.js")();
  9. const Accessor = require("./AccessFlags");
  10. const SML_MAIN=0;
  11. const SML_METH=1;
  12. const SML_ANNO=2;
  13. const SML_PSWITCH=3;
  14. const ERR_PARSE=0;
  15. const LOG_DBG = false;
  16. var LEX = CONST.LEX;
  17. // todo voir WABT, square attack, mesh
  18. var LOG = {
  19. DEBUG: function(txt){
  20. if(LOG_DBG) console.log(txt);
  21. }
  22. };
  23. // end of core class
  24. var Checker = {
  25. isBasicType: function(c){
  26. return (CONST.TYPES[c] != undefined);
  27. },
  28. isObjectType: function(c){
  29. return c=="L";
  30. },
  31. makeFnHashcode: function(modif,cls,name,args,ret){
  32. let xargs = "";
  33. for(let i in args) xargs+="<"+args[i]._hashcode+">";
  34. return modif._name+"|"+cls.name+"|"+name+"|"+xargs+"|"+ret._hashcode;
  35. },
  36. makeFieldHashcode: function(modif,cls,name,type){
  37. //console.log(type);
  38. return modif._name+"|"+cls.name+"|"+name+"|"+type._hashcode;
  39. }
  40. };
  41. /**
  42. * Represent the Smali parser
  43. * @class
  44. */
  45. class SmaliParser
  46. {
  47. constructor(context=null){
  48. this.ctx = context;
  49. this.state = null; // state of the parser
  50. this.subject = null; // parsed smali file
  51. this.obj = null;
  52. this.objReady = false;
  53. this.__tmp_meth = null;
  54. this.__tmp_block = null;
  55. this.__instr_ctr = null;
  56. this.__instr_line = null;
  57. this.currentLine = null;
  58. let self = this;
  59. this.__appendBlock_callback = {
  60. // disabled for perform reasons
  61. /*basicblock: function(meth,block){
  62. this.ctx.bus.send(new Event.Event({
  63. type: "disass.bb.new"
  64. }));
  65. },*/
  66. datablock: function(meth,block){
  67. self.ctx.bus.send(new Event.Event({
  68. type: "disass.datablock.new",
  69. data: block
  70. }));
  71. }
  72. };
  73. }
  74. setContext(context){
  75. this.ctx = context;
  76. }
  77. isModifier(name){
  78. for(let i in LEX.MODIFIER)
  79. if(LEX.MODIFIER[i]==name)
  80. return true;
  81. return false;
  82. }
  83. modifier(src){
  84. if(src instanceof String) src=src.split(LEX.TOKEN.SPACE);
  85. let mod = new Accessor.AccessFlags() , next=true;
  86. //if(src.length<2) return ERR_PARSE;
  87. for(let i=0; i<src.length && next; i++){
  88. mod._match++;
  89. switch(ut.trim(src[i])){
  90. case LEX.MODIFIER.PRIVATE:
  91. mod.private = true;
  92. break;
  93. case LEX.MODIFIER.PROTECTED:
  94. mod.protected = true;
  95. break;
  96. case LEX.MODIFIER.PUBLIC:
  97. mod.public = true;
  98. break;
  99. case LEX.MODIFIER.STATIC:
  100. mod.static = true;
  101. break;
  102. case LEX.MODIFIER.VOLATILE:
  103. mod.volatile = true;
  104. break;
  105. case LEX.MODIFIER.ABSTRACT:
  106. mod.abstract = true;
  107. break;
  108. case LEX.MODIFIER.FINAL:
  109. mod.final = true;
  110. break;
  111. case LEX.MODIFIER.CONSTR:
  112. mod.construct = true;
  113. break;
  114. case LEX.MODIFIER.SYNTHETIC:
  115. mod.synth = true;
  116. break;
  117. case LEX.MODIFIER.ENUM:
  118. mod.enum = true;
  119. break;
  120. case LEX.MODIFIER.TRANSIENT:
  121. mod.transient = true;
  122. break;
  123. case LEX.MODIFIER.DECLSYNC:
  124. mod.declsync = true;
  125. break;
  126. case LEX.MODIFIER.BRIDGE:
  127. mod.bridge = true;
  128. break;
  129. case LEX.MODIFIER.VARARG:
  130. mod.varargs = true;
  131. break;
  132. case LEX.MODIFIER.NATIVE:
  133. mod.native = true;
  134. break;
  135. case LEX.MODIFIER.INTERFACE:
  136. mod.interface = true;
  137. break;
  138. case LEX.MODIFIER.ANNOTATION:
  139. mod.annotation = true;
  140. break;
  141. case LEX.MODIFIER.STRICTFP:
  142. mod.strictfp = true;
  143. break;
  144. case LEX.MODIFIER.SYNCHRONIZED:
  145. mod.synchronized = true;
  146. break;
  147. default:
  148. next=false;
  149. mod._match--;
  150. break;
  151. }
  152. }
  153. if(LOG_DBG){
  154. LOG.DEBUG("[parser::modifier] "+mod.sprint());
  155. }
  156. return mod;
  157. }
  158. fqcn(src){
  159. if(src.length==0) return ERR_PARSE;
  160. let raw="";
  161. raw = (src instanceof Array)? src[0] : src;
  162. // remove additional chars : "L" at begin and ";" at end.
  163. // console.log("PARSER::FQCN > ",src);
  164. let s=raw.substr(1,raw.length-2);
  165. while(s.indexOf("/")>-1) s=s.replace("/",".");
  166. LOG.DEBUG("[parser::fqcn] "+s);
  167. return s;
  168. }
  169. fspath(src){
  170. let s=src;
  171. s = s.substr(s.indexOf(LEX.TOKEN.DELIMITER)+1);
  172. s = s.substr(0,s.indexOf(LEX.TOKEN.DELIMITER));
  173. LOG.DEBUG("[parser::fspath] "+s);
  174. return s;
  175. }
  176. // char
  177. basicTypes(c){
  178. return CONST.TYPES[c];
  179. }
  180. class(src){
  181. let fqcn=null,end=-1;
  182. LOG.DEBUG("---------------------------------------------\n[parser::class] Start ");
  183. //console.log("[?] Instruction parsed : "+OPCODE.CTR);
  184. if(src instanceof String)
  185. src=ut.trim(src).split(LEX.STRUCT.SPACE);
  186. if(src[0]==LEX.STRUCT.CLASS) src.shift();
  187. this.obj = new CLASS.Class();
  188. //console.log(src);
  189. // parse modifiers
  190. this.obj.modifiers = this.modifier(src);
  191. //console.log(src);
  192. // clean src with identified modifier
  193. for(let i=0; i<this.obj.modifiers._match; i++) src.shift();
  194. // console.log(src);
  195. // parse nam
  196. fqcn = this.fqcn(src);
  197. end = fqcn.lastIndexOf(".");
  198. this.obj.fqcn = this.obj.name = fqcn;
  199. this.obj.package = fqcn.substr(0,end);
  200. this.obj.simpleName = fqcn.substr(end+1);
  201. if(this.obj.name.indexOf(LEX.TOKEN.INNER_FQCN)>-1){
  202. this.obj.simpleName = this.obj.simpleName.substr(this.obj.simpleName.indexOf(LEX.TOKEN.INNER_FQCN)+1);
  203. this.obj.innerClass = true;
  204. this.obj.enclosingClass = this.obj.name.substr(0,this.obj.name.indexOf(LEX.TOKEN.INNER_FQCN));
  205. }
  206. this.obj._hashcode = this.obj.hashCode();
  207. LOG.DEBUG("[parser::class] End\n---------------------------------------------");
  208. return this.obj;
  209. }
  210. type(src){
  211. let i=0,l=-1,types=[],s=src,fqn=null,isArray=false;
  212. while(i<src.length){
  213. if(src[i]==LEX.TOKEN.ARRAY){
  214. isArray=true;
  215. i++;
  216. continue;
  217. }
  218. if(src[i]==LEX.TOKEN.OBJREF){
  219. l=src.indexOf(";",i);
  220. fqn=this.fqcn(src.substr(i,l-i+1));
  221. //console.log(fqn);
  222. types.push(new CLASS.ObjectType(fqn, isArray));
  223. i=l+1;
  224. isArray=false;
  225. continue;
  226. }else if(Checker.isBasicType(src[i])){
  227. types.push(new CLASS.BasicType(src[i], isArray));
  228. i++;
  229. isArray = false;
  230. continue;
  231. }
  232. else{
  233. console.log("[!] Unknow type : "+src[i]+" (in "+src+")");
  234. break;
  235. }
  236. }
  237. return types;
  238. }
  239. /**
  240. * To parse a method header
  241. */
  242. methodHeader(src){
  243. let mod = this.modifier(src), raw=null, tmp=null, args=null, ret=null, sa=0, ea=0;
  244. let argTypes = null;
  245. // clean src with identified modifier
  246. for(let i=0; i<mod._match; i++) src.shift();
  247. if(src.length > 1){
  248. console.log(src,mod);
  249. console.log("[!] Method has more modifiers");
  250. }
  251. this.__tmp_meth.modifiers = mod;
  252. raw = ut.trim(src[src.length-1]);
  253. // risque d'UTF8 / autre dans le nom, quid des regexp;
  254. tmp = raw.substr(0,sa=raw.indexOf(LEX.TOKEN.METH_ARG_B));
  255. this.__tmp_meth.name = tmp;
  256. args = raw.substr(sa+1,(ea=raw.indexOf(LEX.TOKEN.METH_ARG_E))-sa-1)
  257. argTypes = this.type(ut.trim(args));
  258. this.__tmp_meth.args = argTypes;
  259. this.__tmp_meth.argsNb = this.__tmp_meth.args.length;
  260. ret=raw.substr(ea+1);
  261. ret = this.type(ut.trim(ret))
  262. if(ret.length == 0)
  263. console.log("[!] this.method error : return type of '"+tmp+"("+args+")' cannot be parsed.")
  264. //exit(0);
  265. this.__tmp_meth.enclosingClass = this.obj;
  266. this.__tmp_meth.ret = ret[0];
  267. this.__tmp_meth._hashcode = this.__tmp_meth.hashCode();
  268. }
  269. instr(src, raw_src, src_line){
  270. let inst = null;//new Instruction();
  271. inst = OPCODE.parse(src,raw_src, src_line);
  272. if(inst != null){
  273. //console.log(inst);
  274. }
  275. //inst.operands = inst.opcode.parse(src);
  276. //console.log('"'+src[0]+'"',inst.opcode);
  277. // todo
  278. return inst;
  279. }
  280. field(src_arr, src_line){
  281. let f=new CLASS.Field(), type=null, tmp=null;
  282. // parse modifiers
  283. f.modifiers = this.modifier(src_arr);
  284. //console.log(f.modifiers);
  285. // clean src with identified modifier
  286. for(let i=0; i<f.modifiers._match; i++) src_arr.shift();
  287. // parse name and type
  288. tmp=src_arr[0].split(":");
  289. f.name=tmp[0];
  290. type=this.type(tmp[1]);
  291. if(type.length>0) f.type=type[0];
  292. //console.log(type.type[0]._hashcode);
  293. f.enclosingClass = this.obj;
  294. f._hashcode = f.hashCode();//Checker.makeFieldHashcode(f.modifiers,this.obj,f.name,f.type);
  295. f.signature();
  296. f.oline = src_line;
  297. LOG.DEBUG("[parser::field] Hashcode : "+f._hashcode);
  298. src_arr.shift();
  299. // parse value if available
  300. if(src_arr.length>0){
  301. // TODO : parse value
  302. f.value = src_arr.pop();
  303. }
  304. return f;
  305. }
  306. method(src, raw_src, src_line){
  307. if(this.state != SML_METH) return null;
  308. let sml=src, hdl=null, catchStmt=null, tmp=null;
  309. switch(ut.trim(sml[0])){
  310. case LEX.STRUCT.METHOD_BEG:
  311. LOG.DEBUG("---------------------------------------------\n[parser::method] Start ");
  312. this.__tmp_meth = new CLASS.Method();
  313. sml.shift();
  314. this.methodHeader(sml,src_line);
  315. this.__tmp_meth.__$in_annot = false;
  316. this.__tmp_block = new CLASS.BasicBlock();
  317. this.__instr_ctr = 0;
  318. break;
  319. case LEX.STRUCT.LOCALS:
  320. this.__tmp_meth.locals = parseInt(sml[1],10);
  321. break;
  322. case LEX.STRUCT.PARAMS:
  323. // this.__tmp_meth.params = parseInt(sml[1],10);
  324. this.__tmp_meth.params.push(
  325. OPCODE.parseParam(sml[1],raw_src)
  326. );
  327. break;
  328. case LEX.STRUCT.REG:
  329. this.__tmp_meth.registers = parseInt(sml[1],10);
  330. break;
  331. case ".prologue":
  332. break;
  333. case LEX.STRUCT.LINE:
  334. // .line is just a metadata associated to an instruction
  335. /*if(this.__tmp_block != null && this.__tmp_block.stack.length > 0){
  336. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  337. this.__tmp_block = new CLASS.BasicBlock();
  338. }*/
  339. // && this.__tmp_block.line != null
  340. this.__tmp_block.line = parseInt(sml[1],10);
  341. this.currentLine = parseInt(sml[1],10);
  342. // source line number
  343. this.__tmp_block.srcln = parseInt(sml[1],10);
  344. break;
  345. case LEX.STRUCT.PSWITCH:
  346. if(sml[1] != undefined){
  347. this.__tmp_block.setupPackedSwitchStatement(parseInt(sml[1],16));
  348. }
  349. break;
  350. case LEX.STRUCT.SSWITCH:
  351. this.__tmp_block.setupSparseSwitchStatement();
  352. break;
  353. case LEX.STRUCT.ARRAY:
  354. this.__tmp_block.setDataWidth(parseInt(sml[1],10));
  355. break;
  356. case LEX.STRUCT.END:
  357. if(sml[1]!=undefined && sml[1]==LEX.STRUCT.METHOD_NAME){
  358. //hdl = this.__tmp_meth._hashcode;
  359. this.state=SML_MAIN;
  360. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  361. this.obj.methods[ this.__tmp_meth.signature()] = this.__tmp_meth;
  362. this.obj._methCount++;
  363. LOG.DEBUG("[parser::method] End\n---------------------------------------------");
  364. }else if(sml[1]!=undefined && sml[1]==LEX.STRUCT.ANNOTATION_NAME){
  365. this.__tmp_meth.__$in_annot = false;
  366. }/*
  367. else if(sml[1]!=undefined && sml[1]==LEX.STRUCT.PSWITCH_NAME){
  368. // nothing to do
  369. //console.log("End of packed switch");
  370. }xs
  371. else if(sml[1]!=undefined && sml[1]==LEX.STRUCT.ARRAY_NAME){
  372. }*/
  373. break;
  374. case LEX.STRUCT.ANNOT_BEG:
  375. // ignore
  376. this.__tmp_meth.__$in_annot = true;
  377. break;
  378. default:
  379. if(this.__tmp_meth.__$in_annot){
  380. // ignore
  381. break;
  382. }
  383. if(sml[0].indexOf(':cond_')>-1){
  384. if(this.__tmp_block instanceof CLASS.DataBlock || this.__tmp_block.stack.length>0){
  385. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  386. this.__tmp_block = new CLASS.BasicBlock();
  387. }
  388. //this.__tmp_block.tag = sml[0];
  389. this.__tmp_block.setAsConditionalBlock(sml[0].split('_')[1]);
  390. }else if(sml[0].indexOf(':goto_')>-1){
  391. if(this.__tmp_block instanceof CLASS.DataBlock || this.__tmp_block.stack.length>0){
  392. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  393. this.__tmp_block = new CLASS.BasicBlock();
  394. }
  395. //this.__tmp_block.tag = sml[0];
  396. this.__tmp_block.setAsGotoBlock(sml[0].split('_')[1]);
  397. }
  398. else if(sml[0].indexOf(':try_start')>-1){
  399. if(this.__tmp_block instanceof CLASS.DataBlock || this.__tmp_block.stack.length>0){
  400. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  401. this.__tmp_block = new CLASS.BasicBlock();
  402. }
  403. // this.__tmp_block.tag = sml[0];
  404. this.__tmp_block.setAsTryBlock(sml[0]);
  405. }
  406. else if(sml[0].indexOf(':try_end')>-1){
  407. this.__tmp_block.setTryEndName(sml[0]);
  408. if(this.__tmp_block != null){
  409. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  410. this.__tmp_block = new CLASS.BasicBlock();
  411. }
  412. }
  413. // >-1
  414. else if(sml[0].indexOf(LEX.LABEL.PSWITCH_DATA)==0 || sml[0].indexOf(LEX.LABEL.SSWITCH_DATA)==0){
  415. if(this.__tmp_block != null){
  416. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  417. this.__tmp_block = new CLASS.BasicBlock();
  418. }
  419. this.__tmp_block.setAsSwitchStatement(sml[0]);
  420. }
  421. // >-1
  422. else if(sml[0].indexOf(LEX.LABEL.PSWITCH)==0){
  423. if(this.__tmp_block.isSwitchStatement()){
  424. this.__tmp_block.switch.appendCase(sml[0]);
  425. }else{
  426. this.__tmp_block.setAsSwitchCase(sml[0]);
  427. }
  428. }
  429. // >-1
  430. else if(sml[0].indexOf(LEX.LABEL.SSWITCH)==0){
  431. if(this.__tmp_block != null
  432. && (this.__tmp_block instanceof CLASS.DataBlock
  433. || this.__tmp_block.stack.length > 0)){
  434. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  435. this.__tmp_block = new CLASS.BasicBlock();
  436. }
  437. this.__tmp_block.setAsSwitchCase(sml[0]);
  438. }
  439. else if(sml.length > 2 && sml[2].indexOf(LEX.LABEL.SSWITCH)>-1 && sml[0].indexOf("p")==-1){
  440. if(this.__tmp_block.isSwitchStatement()){
  441. // console.log(sml);
  442. this.__tmp_block.switch.appendCase(sml[0],sml[2]);
  443. }
  444. }
  445. else if(sml[0].indexOf(LEX.LABEL.ARRAY)>-1){
  446. // check if tmp block not empty (data or bb)
  447. if(this.__tmp_block != null){
  448. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  449. }
  450. this.__tmp_block = new CLASS.DataBlock();
  451. this.__tmp_block.name = sml[0];
  452. }
  453. else if(sml[0].indexOf(LEX.STRUCT.CATCH_ALL)>-1){
  454. catchStmt = new CLASS.CatchStatement();
  455. catchStmt.setTryStart(sml[1].substr(1,sml[1].length));
  456. catchStmt.setTryEnd(sml[3].substr(0,sml[3].length-1));
  457. catchStmt.setTarget(sml[4]);
  458. tmp = this.__tmp_meth.getBasicBlocks();
  459. tmp[tmp.length-1].addCatchStatement(catchStmt);
  460. //this.__tmp_block.addCatchStatement(catchStmt);
  461. }
  462. else if(sml[0].indexOf(LEX.STRUCT.CATCH)>-1){
  463. catchStmt = new CLASS.CatchStatement();
  464. catchStmt.setException(this.type(sml[1])[0]);
  465. catchStmt.setTryStart(sml[2].substr(1,sml[2].length));
  466. catchStmt.setTryEnd(sml[4].substr(0,sml[4].length-1));
  467. catchStmt.setTarget(sml[5]);
  468. //console.log(sml);
  469. tmp = this.__tmp_meth.getBasicBlocks();
  470. tmp[tmp.length-1].addCatchStatement(catchStmt);
  471. /*if(this.__tmp_block != null){
  472. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  473. this.__tmp_block = new CLASS.BasicBlock();
  474. }
  475. this.__tmp_block.addCatchStatement(catchStmt);*/
  476. }
  477. else if(sml[0].indexOf(LEX.LABEL.CATCH)>-1){
  478. if(this.__tmp_block != null
  479. && ( this.__tmp_block instanceof CLASS.DataBlock
  480. || this.__tmp_block.stack.length > 0 )){
  481. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  482. this.__tmp_block = new CLASS.BasicBlock();
  483. }
  484. this.__tmp_block.setCatchCond(raw_src);
  485. }
  486. else if(sml[0].indexOf(':catchall')>-1){
  487. if(this.__tmp_block != null
  488. && ( this.__tmp_block instanceof CLASS.DataBlock
  489. || this.__tmp_block.stack.length > 0 )){
  490. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  491. this.__tmp_block = new CLASS.BasicBlock();
  492. }
  493. this.__tmp_block.setAsCatchBlock(sml[0]);
  494. }
  495. else if(this.__tmp_block instanceof CLASS.DataBlock){
  496. hdl = CONST.RE.ARRAY_VALUE.exec(sml[0]);
  497. if(hdl ==null) break;
  498. this.__tmp_block.pushData("0x"+hdl[2], (hdl[1] != undefined), (hdl[3]=='s'));
  499. }
  500. else{
  501. if(this.__tmp_block instanceof CLASS.DataBlock) console.log("Error : DataBlock instead of BasicBlock",this.__tmp_meth);
  502. if(this.__tmp_block == null) console.log("Error : tmpBlock is null",this.__tmp_meth);
  503. hdl = this.instr(sml,raw_src,src_line);
  504. if(hdl !== null){
  505. this.__instr_ctr++;
  506. hdl.offset = this.__instr_ctr;
  507. hdl.oline = this.__instr_ctr;
  508. if(this.currentLine != null){
  509. hdl.iline = this.currentLine;
  510. this.currentLine = null;
  511. }
  512. this.__tmp_block.stack.push(hdl);
  513. if(hdl.opcode.type == CONST.INSTR_TYPE.IF || hdl.opcode.type == CONST.INSTR_TYPE.GOTO ) {
  514. if(this.__tmp_block instanceof CLASS.DataBlock || this.__tmp_block.stack.length>0){
  515. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  516. this.__tmp_block = new CLASS.BasicBlock();
  517. }
  518. }
  519. }
  520. // add end of basic block on If / Goto
  521. /* else if(this.isJumpInstruction(sml)){
  522. if(this.__tmp_block instanceof CLASS.DataBlock || this.__tmp_block.stack.length>0){
  523. this.__tmp_meth.appendBlock(this.__tmp_block, this.__appendBlock_callback);
  524. this.__tmp_block = new CLASS.BasicBlock();
  525. }
  526. if(this.__tmp_block == null) console.log("Error : tmpBlock is null",this.__tmp_meth);
  527. hdl = this.instr(sml,raw_src,src_line);
  528. if(hdl !== null){
  529. this.__instr_ctr++;
  530. hdl.offset = this.__instr_ctr;
  531. hdl.oline = this.__instr_ctr;
  532. this.__tmp_block.stack.push(hdl);
  533. }
  534. //this.__tmp_block.tag = sml[0];
  535. //this.__tmp_block.setAsGotoBlock(sml[0].split('_')[1]);
  536. }*/
  537. }
  538. break;
  539. }
  540. return true;
  541. }
  542. annotation(src){
  543. if(this.state != SML_ANNO) return null;
  544. let sml=[], hdl=null;
  545. sml=(src instanceof String)? src.split(LEX.TOKEN.SPACE) : src ;
  546. // search lexeme
  547. switch(ut.trim(sml[0])){
  548. case LEX.STRUCT.END:
  549. if(sml[1]!=undefined && sml[1]=="annotation"){
  550. this.state=SML_MAIN;
  551. }
  552. break;
  553. }
  554. //console.log("[!] this.annotation not implemented");
  555. }
  556. parse(src){
  557. let ls=src.split("\n"), ln=null, sml=null, obj=null;
  558. //console.log(ls);
  559. for(let l=0; l<ls.length; l++){
  560. ln=ut.trim(ls[l]);
  561. if(ln.length==0){
  562. continue;
  563. }
  564. sml=ln.split(LEX.TOKEN.SPACE);
  565. switch(sml[0]){
  566. case LEX.STRUCT.CLASS:
  567. sml.shift();
  568. this.class(sml);
  569. break;
  570. case LEX.STRUCT.IMPLEMENTS:
  571. sml.shift();
  572. this.obj.implements.push(this.fqcn(sml[0]));
  573. break;
  574. case LEX.STRUCT.SUPER:
  575. sml.shift();
  576. this.obj.extends = this.fqcn(sml[0]);
  577. break;
  578. case LEX.STRUCT.SRC:
  579. this.obj.source = this.fspath(sml[1]);
  580. break;
  581. case LEX.STRUCT.FIELD:
  582. sml.shift();
  583. obj=this.field(sml,l);
  584. // use an internal name which combine visibility and field name
  585. //this.obj.fields[obj._hashcode] = obj;
  586. this.obj.fields[obj.signature()] = obj;
  587. this.obj._fieldCount++;
  588. break;
  589. case LEX.STRUCT.METHOD_BEG:
  590. this.state = SML_METH;
  591. this.method(sml,ln,l);
  592. break;
  593. case LEX.STRUCT.ANNOT_BEG:
  594. if(this.state != SML_METH){
  595. this.state = SML_ANNO;
  596. this.annotation(sml);
  597. }
  598. break;
  599. default:
  600. switch(this.state){
  601. case SML_METH:
  602. this.method(sml,ln,l);
  603. break;
  604. case SML_PSWITCH:
  605. this.pswitch(sml,ln,l);
  606. break;
  607. case SML_ANNO:
  608. this.annotation(sml);
  609. break;
  610. }
  611. break;
  612. }
  613. }
  614. //console.log(this.obj);
  615. //this.obj.dump();
  616. return this.obj;
  617. }
  618. parseStream( pFilePath, pEncoding, pCallback){
  619. let _self = this, rs=null, stream=null;
  620. _self.obj = null;
  621. _self.objReady = false;
  622. stream =_fs_.createReadStream(pFilePath)
  623. .pipe(_es_.split())
  624. .pipe(_es_.mapSync(function(line){
  625. // pause the readstream
  626. stream.pause();
  627. // process line here and call s.resume() when rdy
  628. // function below was for logging memory usage
  629. console.log(line);
  630. // resume the readstream, possibly from a callback
  631. stream.resume();
  632. })
  633. .on('error', function(err){
  634. console.log('Error while reading file.', err);
  635. })
  636. .on('end', function(){
  637. console.log('Read entire file.')
  638. })
  639. );
  640. //stream = _fs_.createReadStream(pFilePath,{ encoding: 'utf8' });
  641. //stream = LineStream(stream);
  642. /*
  643. stream.on('resume', function(){
  644. console.log('resume');
  645. });*/
  646. /*
  647. stream.on('pause', function(){
  648. console.log('paused');
  649. });
  650. stream.on('close', function(){
  651. console.log("close", _self.obj)
  652. pCallback(_self.obj);
  653. });
  654. stream.on('data', function(pLine){
  655. let ln=null, sml=null, obj=null;
  656. console.log(pLine);
  657. ln=ut.trim(pLine);
  658. if(ln.length==0){
  659. return;
  660. }
  661. sml=ln.split(LEX.TOKEN.SPACE);
  662. switch(sml[0]){
  663. case LEX.STRUCT.CLASS:
  664. sml.shift();
  665. _self.class(sml);
  666. break;
  667. case LEX.STRUCT.IMPLEMENTS:
  668. sml.shift();
  669. _self.obj.implements.push(_self.fqcn(sml[0]));
  670. break;
  671. case LEX.STRUCT.SUPER:
  672. sml.shift();
  673. _self.obj.extends = _self.fqcn(sml[0]);
  674. break;
  675. case LEX.STRUCT.SRC:
  676. _self.obj.source = _self.fspath(sml[1]);
  677. break;
  678. case LEX.STRUCT.FIELD:
  679. sml.shift();
  680. obj=_self.field(sml,l);
  681. // use an internal name which combine visibility and field name
  682. //this.obj.fields[obj._hashcode] = obj;
  683. _self.obj.fields[obj.signature()] = obj;
  684. _self.obj._fieldCount++;
  685. break;
  686. case LEX.STRUCT.METHOD_BEG:
  687. _self.state = SML_METH;
  688. _self.method(sml,ln,l);
  689. break;
  690. case LEX.STRUCT.ANNOT_BEG:
  691. if(_self.state != SML_METH){
  692. _self.state = SML_ANNO;
  693. _self.annotation(sml);
  694. }
  695. break;
  696. default:
  697. switch(_self.state){
  698. case SML_METH:
  699. _self.method(sml,ln,l);
  700. break;
  701. case SML_PSWITCH:
  702. _self.pswitch(sml,ln,l);
  703. break;
  704. case SML_ANNO:
  705. _self.annotation(sml);
  706. break;
  707. }
  708. break;
  709. }
  710. });*/
  711. }
  712. }
  713. module.exports = SmaliParser;