expr.cpp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677
  1. #include "pocketpy/expr.h"
  2. namespace pkpy{
  3. int CodeEmitContext::get_loop() const {
  4. int index = curr_block_i;
  5. while(index >= 0){
  6. if(co->blocks[index].type == FOR_LOOP) break;
  7. if(co->blocks[index].type == WHILE_LOOP) break;
  8. index = co->blocks[index].parent;
  9. }
  10. return index;
  11. }
  12. CodeBlock* CodeEmitContext::enter_block(CodeBlockType type){
  13. if(type == FOR_LOOP) for_loop_depth++;
  14. co->blocks.push_back(CodeBlock(
  15. type, curr_block_i, for_loop_depth, (int)co->codes.size()
  16. ));
  17. curr_block_i = co->blocks.size()-1;
  18. return &co->blocks[curr_block_i];
  19. }
  20. void CodeEmitContext::exit_block(){
  21. auto curr_type = co->blocks[curr_block_i].type;
  22. if(curr_type == FOR_LOOP) for_loop_depth--;
  23. co->blocks[curr_block_i].end = co->codes.size();
  24. curr_block_i = co->blocks[curr_block_i].parent;
  25. if(curr_block_i < 0) PK_FATAL_ERROR();
  26. if(curr_type == FOR_LOOP){
  27. // add a no op here to make block check work
  28. emit_(OP_NO_OP, BC_NOARG, BC_KEEPLINE);
  29. }
  30. }
  31. // clear the expression stack and generate bytecode
  32. void CodeEmitContext::emit_expr(){
  33. if(s_expr.size() != 1){
  34. throw std::runtime_error("s_expr.size() != 1\n" + _log_s_expr());
  35. }
  36. Expr_ expr = s_expr.popx();
  37. expr->emit_(this);
  38. }
  39. std::string CodeEmitContext::_log_s_expr(){
  40. std::stringstream ss; // debug
  41. for(auto& e: s_expr.data()) ss << e->str() << " ";
  42. return ss.str();
  43. }
  44. int CodeEmitContext::emit_(Opcode opcode, uint16_t arg, int line) {
  45. co->codes.push_back(Bytecode{(uint8_t)opcode, arg});
  46. co->iblocks.push_back(curr_block_i);
  47. co->lines.push_back(line);
  48. int i = co->codes.size() - 1;
  49. if(line==BC_KEEPLINE){
  50. if(i>=1) co->lines[i] = co->lines[i-1];
  51. else co->lines[i] = 1;
  52. }
  53. return i;
  54. }
  55. void CodeEmitContext::patch_jump(int index) {
  56. int target = co->codes.size();
  57. co->codes[index].arg = target;
  58. }
  59. bool CodeEmitContext::add_label(StrName name){
  60. if(co->labels.contains(name)) return false;
  61. co->labels.set(name, co->codes.size());
  62. return true;
  63. }
  64. int CodeEmitContext::add_varname(StrName name){
  65. // PK_MAX_CO_VARNAMES will be checked when pop_context(), not here
  66. int index = co->varnames_inv.try_get(name);
  67. if(index >= 0) return index;
  68. co->varnames.push_back(name);
  69. index = co->varnames.size() - 1;
  70. co->varnames_inv.set(name, index);
  71. return index;
  72. }
  73. int CodeEmitContext::add_const(PyObject* v){
  74. // simple deduplication, only works for int/float
  75. for(int i=0; i<co->consts.size(); i++){
  76. if(co->consts[i] == v) return i;
  77. }
  78. // string deduplication
  79. if(is_non_tagged_type(v, vm->tp_str)){
  80. const Str& v_str = PK_OBJ_GET(Str, v);
  81. for(int i=0; i<co->consts.size(); i++){
  82. if(is_non_tagged_type(co->consts[i], vm->tp_str)){
  83. if(PK_OBJ_GET(Str, co->consts[i]) == v_str) return i;
  84. }
  85. }
  86. }
  87. co->consts.push_back(v);
  88. return co->consts.size() - 1;
  89. }
  90. int CodeEmitContext::add_func_decl(FuncDecl_ decl){
  91. co->func_decls.push_back(decl);
  92. return co->func_decls.size() - 1;
  93. }
  94. void CodeEmitContext::emit_store_name(NameScope scope, StrName name, int line){
  95. switch(scope){
  96. case NAME_LOCAL:
  97. emit_(OP_STORE_FAST, add_varname(name), line);
  98. break;
  99. case NAME_GLOBAL:
  100. emit_(OP_STORE_GLOBAL, StrName(name).index, line);
  101. break;
  102. case NAME_GLOBAL_UNKNOWN:
  103. emit_(OP_STORE_NAME, StrName(name).index, line);
  104. break;
  105. default: PK_FATAL_ERROR(); break;
  106. }
  107. }
  108. void NameExpr::emit_(CodeEmitContext* ctx) {
  109. int index = ctx->co->varnames_inv.try_get(name);
  110. if(scope == NAME_LOCAL && index >= 0){
  111. ctx->emit_(OP_LOAD_FAST, index, line);
  112. }else{
  113. Opcode op = ctx->level <= 1 ? OP_LOAD_GLOBAL : OP_LOAD_NONLOCAL;
  114. if(ctx->is_compiling_class && scope == NAME_GLOBAL){
  115. // if we are compiling a class, we should use OP_LOAD_ATTR_GLOBAL instead of OP_LOAD_GLOBAL
  116. // this supports @property.setter
  117. op = OP_LOAD_CLASS_GLOBAL;
  118. // exec()/eval() won't work with OP_LOAD_ATTR_GLOBAL in class body
  119. }else{
  120. // we cannot determine the scope when calling exec()/eval()
  121. if(scope == NAME_GLOBAL_UNKNOWN) op = OP_LOAD_NAME;
  122. }
  123. ctx->emit_(op, StrName(name).index, line);
  124. }
  125. }
  126. bool NameExpr::emit_del(CodeEmitContext* ctx) {
  127. switch(scope){
  128. case NAME_LOCAL:
  129. ctx->emit_(OP_DELETE_FAST, ctx->add_varname(name), line);
  130. break;
  131. case NAME_GLOBAL:
  132. ctx->emit_(OP_DELETE_GLOBAL, StrName(name).index, line);
  133. break;
  134. case NAME_GLOBAL_UNKNOWN:
  135. ctx->emit_(OP_DELETE_NAME, StrName(name).index, line);
  136. break;
  137. default: PK_FATAL_ERROR(); break;
  138. }
  139. return true;
  140. }
  141. bool NameExpr::emit_store(CodeEmitContext* ctx) {
  142. if(ctx->is_compiling_class){
  143. ctx->emit_(OP_STORE_CLASS_ATTR, name.index, line);
  144. return true;
  145. }
  146. ctx->emit_store_name(scope, name, line);
  147. return true;
  148. }
  149. void InvertExpr::emit_(CodeEmitContext* ctx) {
  150. child->emit_(ctx);
  151. ctx->emit_(OP_UNARY_INVERT, BC_NOARG, line);
  152. }
  153. void StarredExpr::emit_(CodeEmitContext* ctx) {
  154. child->emit_(ctx);
  155. ctx->emit_(OP_UNARY_STAR, level, line);
  156. }
  157. bool StarredExpr::emit_store(CodeEmitContext* ctx) {
  158. if(level != 1) return false;
  159. // simply proxy to child
  160. return child->emit_store(ctx);
  161. }
  162. void NotExpr::emit_(CodeEmitContext* ctx) {
  163. child->emit_(ctx);
  164. ctx->emit_(OP_UNARY_NOT, BC_NOARG, line);
  165. }
  166. void AndExpr::emit_(CodeEmitContext* ctx) {
  167. lhs->emit_(ctx);
  168. int patch = ctx->emit_(OP_JUMP_IF_FALSE_OR_POP, BC_NOARG, line);
  169. rhs->emit_(ctx);
  170. ctx->patch_jump(patch);
  171. }
  172. void OrExpr::emit_(CodeEmitContext* ctx) {
  173. lhs->emit_(ctx);
  174. int patch = ctx->emit_(OP_JUMP_IF_TRUE_OR_POP, BC_NOARG, line);
  175. rhs->emit_(ctx);
  176. ctx->patch_jump(patch);
  177. }
  178. void Literal0Expr::emit_(CodeEmitContext* ctx){
  179. switch (token) {
  180. case TK("None"): ctx->emit_(OP_LOAD_NONE, BC_NOARG, line); break;
  181. case TK("True"): ctx->emit_(OP_LOAD_TRUE, BC_NOARG, line); break;
  182. case TK("False"): ctx->emit_(OP_LOAD_FALSE, BC_NOARG, line); break;
  183. case TK("..."): ctx->emit_(OP_LOAD_ELLIPSIS, BC_NOARG, line); break;
  184. default: PK_FATAL_ERROR();
  185. }
  186. }
  187. void LongExpr::emit_(CodeEmitContext* ctx) {
  188. VM* vm = ctx->vm;
  189. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR(s)), line);
  190. ctx->emit_(OP_BUILD_LONG, BC_NOARG, line);
  191. }
  192. void ImagExpr::emit_(CodeEmitContext* ctx) {
  193. VM* vm = ctx->vm;
  194. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR(value)), line);
  195. ctx->emit_(OP_BUILD_IMAG, BC_NOARG, line);
  196. }
  197. void BytesExpr::emit_(CodeEmitContext* ctx) {
  198. VM* vm = ctx->vm;
  199. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR(s)), line);
  200. ctx->emit_(OP_BUILD_BYTES, BC_NOARG, line);
  201. }
  202. void LiteralExpr::emit_(CodeEmitContext* ctx) {
  203. VM* vm = ctx->vm;
  204. PyObject* obj = nullptr;
  205. if(std::holds_alternative<i64>(value)){
  206. i64 _val = std::get<i64>(value);
  207. if(_val >= INT16_MIN && _val <= INT16_MAX){
  208. ctx->emit_(OP_LOAD_INTEGER, (uint16_t)_val, line);
  209. return;
  210. }
  211. obj = VAR(_val);
  212. }
  213. if(std::holds_alternative<f64>(value)){
  214. obj = VAR(std::get<f64>(value));
  215. }
  216. if(std::holds_alternative<Str>(value)){
  217. obj = VAR(std::get<Str>(value));
  218. }
  219. PK_ASSERT(obj != nullptr)
  220. ctx->emit_(OP_LOAD_CONST, ctx->add_const(obj), line);
  221. }
  222. void NegatedExpr::emit_(CodeEmitContext* ctx){
  223. VM* vm = ctx->vm;
  224. // if child is a int of float, do constant folding
  225. if(child->is_literal()){
  226. LiteralExpr* lit = static_cast<LiteralExpr*>(child.get());
  227. if(std::holds_alternative<i64>(lit->value)){
  228. i64 _val = -std::get<i64>(lit->value);
  229. if(_val >= INT16_MIN && _val <= INT16_MAX){
  230. ctx->emit_(OP_LOAD_INTEGER, (uint16_t)_val, line);
  231. }else{
  232. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR(_val)), line);
  233. }
  234. return;
  235. }
  236. if(std::holds_alternative<f64>(lit->value)){
  237. PyObject* obj = VAR(-std::get<f64>(lit->value));
  238. ctx->emit_(OP_LOAD_CONST, ctx->add_const(obj), line);
  239. return;
  240. }
  241. }
  242. child->emit_(ctx);
  243. ctx->emit_(OP_UNARY_NEGATIVE, BC_NOARG, line);
  244. }
  245. void SliceExpr::emit_(CodeEmitContext* ctx){
  246. if(start){
  247. start->emit_(ctx);
  248. }else{
  249. ctx->emit_(OP_LOAD_NONE, BC_NOARG, line);
  250. }
  251. if(stop){
  252. stop->emit_(ctx);
  253. }else{
  254. ctx->emit_(OP_LOAD_NONE, BC_NOARG, line);
  255. }
  256. if(step){
  257. step->emit_(ctx);
  258. }else{
  259. ctx->emit_(OP_LOAD_NONE, BC_NOARG, line);
  260. }
  261. ctx->emit_(OP_BUILD_SLICE, BC_NOARG, line);
  262. }
  263. void DictItemExpr::emit_(CodeEmitContext* ctx) {
  264. if(is_starred()){
  265. PK_ASSERT(key == nullptr);
  266. value->emit_(ctx);
  267. }else{
  268. value->emit_(ctx);
  269. key->emit_(ctx); // reverse order
  270. ctx->emit_(OP_BUILD_TUPLE, 2, line);
  271. }
  272. }
  273. bool TupleExpr::emit_store(CodeEmitContext* ctx) {
  274. // TOS is an iterable
  275. // items may contain StarredExpr, we should check it
  276. int starred_i = -1;
  277. for(int i=0; i<items.size(); i++){
  278. if(!items[i]->is_starred()) continue;
  279. if(starred_i == -1) starred_i = i;
  280. else return false; // multiple StarredExpr not allowed
  281. }
  282. if(starred_i == -1){
  283. Bytecode& prev = ctx->co->codes.back();
  284. if(prev.op == OP_BUILD_TUPLE && prev.arg == items.size()){
  285. // build tuple and unpack it is meaningless
  286. prev.op = OP_NO_OP;
  287. prev.arg = BC_NOARG;
  288. }else{
  289. ctx->emit_(OP_UNPACK_SEQUENCE, items.size(), line);
  290. }
  291. }else{
  292. // starred assignment target must be in a tuple
  293. if(items.size() == 1) return false;
  294. // starred assignment target must be the last one (differ from cpython)
  295. if(starred_i != items.size()-1) return false;
  296. // a,*b = [1,2,3]
  297. // stack is [1,2,3] -> [1,[2,3]]
  298. ctx->emit_(OP_UNPACK_EX, items.size()-1, line);
  299. }
  300. // do reverse emit
  301. for(int i=items.size()-1; i>=0; i--){
  302. bool ok = items[i]->emit_store(ctx);
  303. if(!ok) return false;
  304. }
  305. return true;
  306. }
  307. bool TupleExpr::emit_del(CodeEmitContext* ctx){
  308. for(auto& e: items){
  309. bool ok = e->emit_del(ctx);
  310. if(!ok) return false;
  311. }
  312. return true;
  313. }
  314. void CompExpr::emit_(CodeEmitContext* ctx){
  315. ctx->emit_(op0(), 0, line);
  316. iter->emit_(ctx);
  317. ctx->emit_(OP_GET_ITER, BC_NOARG, BC_KEEPLINE);
  318. ctx->enter_block(FOR_LOOP);
  319. ctx->emit_(OP_FOR_ITER, BC_NOARG, BC_KEEPLINE);
  320. bool ok = vars->emit_store(ctx);
  321. // this error occurs in `vars` instead of this line, but...nevermind
  322. PK_ASSERT(ok); // TODO: raise a SyntaxError instead
  323. if(cond){
  324. cond->emit_(ctx);
  325. int patch = ctx->emit_(OP_POP_JUMP_IF_FALSE, BC_NOARG, BC_KEEPLINE);
  326. expr->emit_(ctx);
  327. ctx->emit_(op1(), BC_NOARG, BC_KEEPLINE);
  328. ctx->patch_jump(patch);
  329. }else{
  330. expr->emit_(ctx);
  331. ctx->emit_(op1(), BC_NOARG, BC_KEEPLINE);
  332. }
  333. ctx->emit_(OP_LOOP_CONTINUE, ctx->get_loop(), BC_KEEPLINE);
  334. ctx->exit_block();
  335. }
  336. void FStringExpr::_load_simple_expr(CodeEmitContext* ctx, Str expr){
  337. bool repr = false;
  338. if(expr.size>=2 && expr.end()[-2]=='!'){
  339. switch(expr.end()[-1]){
  340. case 'r': repr = true; expr = expr.substr(0, expr.size-2); break;
  341. case 's': repr = false; expr = expr.substr(0, expr.size-2); break;
  342. default: break; // nothing happens
  343. }
  344. }
  345. // name or name.name
  346. PK_LOCAL_STATIC const std::regex pattern(R"(^[a-zA-Z_][a-zA-Z0-9_]*(\.[a-zA-Z_][a-zA-Z0-9_]*){0,1}$)");
  347. if(std::regex_match(expr.str(), pattern)){
  348. int dot = expr.index(".");
  349. if(dot < 0){
  350. ctx->emit_(OP_LOAD_NAME, StrName(expr.sv()).index, line);
  351. }else{
  352. StrName name(expr.substr(0, dot).sv());
  353. StrName attr(expr.substr(dot+1).sv());
  354. ctx->emit_(OP_LOAD_NAME, name.index, line);
  355. ctx->emit_(OP_LOAD_ATTR, attr.index, line);
  356. }
  357. }else{
  358. int index = ctx->add_const(py_var(ctx->vm, expr));
  359. ctx->emit_(OP_FSTRING_EVAL, index, line);
  360. }
  361. if(repr){
  362. ctx->emit_(OP_REPR, BC_NOARG, line);
  363. }
  364. }
  365. void FStringExpr::emit_(CodeEmitContext* ctx){
  366. VM* vm = ctx->vm;
  367. int i = 0; // left index
  368. int j = 0; // right index
  369. int count = 0; // how many string parts
  370. bool flag = false; // true if we are in a expression
  371. const char* fmt_valid_chars = "0-=*#@!~" "<>^" ".fds" "0123456789";
  372. PK_LOCAL_STATIC const std::set<char> fmt_valid_char_set(fmt_valid_chars, fmt_valid_chars + strlen(fmt_valid_chars));
  373. while(j < src.size){
  374. if(flag){
  375. if(src[j] == '}'){
  376. // add expression
  377. Str expr = src.substr(i, j-i);
  378. // BUG: ':' is not a format specifier in f"{stack[2:]}"
  379. int conon = expr.index(":");
  380. if(conon >= 0){
  381. Str spec = expr.substr(conon+1);
  382. // filter some invalid spec
  383. bool ok = true;
  384. for(char c: spec) if(!fmt_valid_char_set.count(c)){ ok = false; break; }
  385. if(ok){
  386. _load_simple_expr(ctx, expr.substr(0, conon));
  387. ctx->emit_(OP_FORMAT_STRING, ctx->add_const(VAR(spec)), line);
  388. }else{
  389. // ':' is not a spec indicator
  390. _load_simple_expr(ctx, expr);
  391. }
  392. }else{
  393. _load_simple_expr(ctx, expr);
  394. }
  395. flag = false;
  396. count++;
  397. }
  398. }else{
  399. if(src[j] == '{'){
  400. // look at next char
  401. if(j+1 < src.size && src[j+1] == '{'){
  402. // {{ -> {
  403. j++;
  404. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR("{")), line);
  405. count++;
  406. }else{
  407. // { -> }
  408. flag = true;
  409. i = j+1;
  410. }
  411. }else if(src[j] == '}'){
  412. // look at next char
  413. if(j+1 < src.size && src[j+1] == '}'){
  414. // }} -> }
  415. j++;
  416. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR("}")), line);
  417. count++;
  418. }else{
  419. // } -> error
  420. // throw std::runtime_error("f-string: unexpected }");
  421. // just ignore
  422. }
  423. }else{
  424. // literal
  425. i = j;
  426. while(j < src.size && src[j] != '{' && src[j] != '}') j++;
  427. Str literal = src.substr(i, j-i);
  428. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR(literal)), line);
  429. count++;
  430. continue; // skip j++
  431. }
  432. }
  433. j++;
  434. }
  435. if(flag){
  436. // literal
  437. Str literal = src.substr(i, src.size-i);
  438. ctx->emit_(OP_LOAD_CONST, ctx->add_const(VAR(literal)), line);
  439. count++;
  440. }
  441. ctx->emit_(OP_BUILD_STRING, count, line);
  442. }
  443. void SubscrExpr::emit_(CodeEmitContext* ctx){
  444. a->emit_(ctx);
  445. b->emit_(ctx);
  446. ctx->emit_(OP_LOAD_SUBSCR, BC_NOARG, line);
  447. }
  448. bool SubscrExpr::emit_del(CodeEmitContext* ctx){
  449. a->emit_(ctx);
  450. b->emit_(ctx);
  451. ctx->emit_(OP_DELETE_SUBSCR, BC_NOARG, line);
  452. return true;
  453. }
  454. bool SubscrExpr::emit_store(CodeEmitContext* ctx){
  455. a->emit_(ctx);
  456. b->emit_(ctx);
  457. ctx->emit_(OP_STORE_SUBSCR, BC_NOARG, line);
  458. return true;
  459. }
  460. void AttribExpr::emit_(CodeEmitContext* ctx){
  461. a->emit_(ctx);
  462. int index = StrName(b).index;
  463. ctx->emit_(OP_LOAD_ATTR, index, line);
  464. }
  465. bool AttribExpr::emit_del(CodeEmitContext* ctx) {
  466. a->emit_(ctx);
  467. int index = StrName(b).index;
  468. ctx->emit_(OP_DELETE_ATTR, index, line);
  469. return true;
  470. }
  471. bool AttribExpr::emit_store(CodeEmitContext* ctx){
  472. a->emit_(ctx);
  473. int index = StrName(b).index;
  474. ctx->emit_(OP_STORE_ATTR, index, line);
  475. return true;
  476. }
  477. void AttribExpr::emit_method(CodeEmitContext* ctx) {
  478. a->emit_(ctx);
  479. int index = StrName(b).index;
  480. ctx->emit_(OP_LOAD_METHOD, index, line);
  481. }
  482. void CallExpr::emit_(CodeEmitContext* ctx) {
  483. bool vargs = false;
  484. bool vkwargs = false;
  485. for(auto& arg: args) if(arg->is_starred()) vargs = true;
  486. for(auto& item: kwargs) if(item.second->is_starred()) vkwargs = true;
  487. // if callable is a AttrExpr, we should try to use `fast_call` instead of use `boundmethod` proxy
  488. if(callable->is_attrib()){
  489. auto p = static_cast<AttribExpr*>(callable.get());
  490. p->emit_method(ctx); // OP_LOAD_METHOD
  491. }else{
  492. callable->emit_(ctx);
  493. ctx->emit_(OP_LOAD_NULL, BC_NOARG, BC_KEEPLINE);
  494. }
  495. if(vargs || vkwargs){
  496. for(auto& item: args) item->emit_(ctx);
  497. ctx->emit_(OP_BUILD_TUPLE_UNPACK, (uint16_t)args.size(), line);
  498. if(!kwargs.empty()){
  499. for(auto& item: kwargs){
  500. if(item.second->is_starred()){
  501. PK_ASSERT(item.second->star_level() == 2)
  502. item.second->emit_(ctx);
  503. }else{
  504. // k=v
  505. int index = ctx->add_const(py_var(ctx->vm, item.first));
  506. ctx->emit_(OP_LOAD_CONST, index, line);
  507. item.second->emit_(ctx);
  508. ctx->emit_(OP_BUILD_TUPLE, 2, line);
  509. }
  510. }
  511. ctx->emit_(OP_BUILD_DICT_UNPACK, (int)kwargs.size(), line);
  512. ctx->emit_(OP_CALL_TP, 1, line);
  513. }else{
  514. ctx->emit_(OP_CALL_TP, 0, line);
  515. }
  516. }else{
  517. // vectorcall protocal
  518. for(auto& item: args) item->emit_(ctx);
  519. for(auto& item: kwargs){
  520. uint16_t index = StrName(item.first.sv()).index;
  521. ctx->emit_(OP_LOAD_INTEGER, index, line);
  522. item.second->emit_(ctx);
  523. }
  524. int KWARGC = kwargs.size();
  525. int ARGC = args.size();
  526. ctx->emit_(OP_CALL, (KWARGC<<8)|ARGC, line);
  527. }
  528. }
  529. bool BinaryExpr::is_compare() const {
  530. switch(op){
  531. case TK("<"): case TK("<="): case TK("=="):
  532. case TK("!="): case TK(">"): case TK(">="): return true;
  533. default: return false;
  534. }
  535. }
  536. void BinaryExpr::_emit_compare(CodeEmitContext* ctx, std::vector<int>& jmps){
  537. if(lhs->is_compare()){
  538. static_cast<BinaryExpr*>(lhs.get())->_emit_compare(ctx, jmps);
  539. }else{
  540. lhs->emit_(ctx); // [a]
  541. }
  542. rhs->emit_(ctx); // [a, b]
  543. ctx->emit_(OP_DUP_TOP, BC_NOARG, line); // [a, b, b]
  544. ctx->emit_(OP_ROT_THREE, BC_NOARG, line); // [b, a, b]
  545. switch(op){
  546. case TK("<"): ctx->emit_(OP_COMPARE_LT, BC_NOARG, line); break;
  547. case TK("<="): ctx->emit_(OP_COMPARE_LE, BC_NOARG, line); break;
  548. case TK("=="): ctx->emit_(OP_COMPARE_EQ, BC_NOARG, line); break;
  549. case TK("!="): ctx->emit_(OP_COMPARE_NE, BC_NOARG, line); break;
  550. case TK(">"): ctx->emit_(OP_COMPARE_GT, BC_NOARG, line); break;
  551. case TK(">="): ctx->emit_(OP_COMPARE_GE, BC_NOARG, line); break;
  552. default: UNREACHABLE();
  553. }
  554. // [b, RES]
  555. int index = ctx->emit_(OP_SHORTCUT_IF_FALSE_OR_POP, BC_NOARG, line);
  556. jmps.push_back(index);
  557. }
  558. void BinaryExpr::emit_(CodeEmitContext* ctx) {
  559. std::vector<int> jmps;
  560. if(is_compare() && lhs->is_compare()){
  561. // (a < b) < c
  562. static_cast<BinaryExpr*>(lhs.get())->_emit_compare(ctx, jmps);
  563. // [b, RES]
  564. }else{
  565. // (1 + 2) < c
  566. lhs->emit_(ctx);
  567. }
  568. rhs->emit_(ctx);
  569. switch (op) {
  570. case TK("+"): ctx->emit_(OP_BINARY_ADD, BC_NOARG, line); break;
  571. case TK("-"): ctx->emit_(OP_BINARY_SUB, BC_NOARG, line); break;
  572. case TK("*"): ctx->emit_(OP_BINARY_MUL, BC_NOARG, line); break;
  573. case TK("/"): ctx->emit_(OP_BINARY_TRUEDIV, BC_NOARG, line); break;
  574. case TK("//"): ctx->emit_(OP_BINARY_FLOORDIV, BC_NOARG, line); break;
  575. case TK("%"): ctx->emit_(OP_BINARY_MOD, BC_NOARG, line); break;
  576. case TK("**"): ctx->emit_(OP_BINARY_POW, BC_NOARG, line); break;
  577. case TK("<"): ctx->emit_(OP_COMPARE_LT, BC_NOARG, line); break;
  578. case TK("<="): ctx->emit_(OP_COMPARE_LE, BC_NOARG, line); break;
  579. case TK("=="): ctx->emit_(OP_COMPARE_EQ, BC_NOARG, line); break;
  580. case TK("!="): ctx->emit_(OP_COMPARE_NE, BC_NOARG, line); break;
  581. case TK(">"): ctx->emit_(OP_COMPARE_GT, BC_NOARG, line); break;
  582. case TK(">="): ctx->emit_(OP_COMPARE_GE, BC_NOARG, line); break;
  583. case TK("in"): ctx->emit_(OP_CONTAINS_OP, 0, line); break;
  584. case TK("not in"): ctx->emit_(OP_CONTAINS_OP, 1, line); break;
  585. case TK("is"): ctx->emit_(OP_IS_OP, 0, line); break;
  586. case TK("is not"): ctx->emit_(OP_IS_OP, 1, line); break;
  587. case TK("<<"): ctx->emit_(OP_BITWISE_LSHIFT, BC_NOARG, line); break;
  588. case TK(">>"): ctx->emit_(OP_BITWISE_RSHIFT, BC_NOARG, line); break;
  589. case TK("&"): ctx->emit_(OP_BITWISE_AND, BC_NOARG, line); break;
  590. case TK("|"): ctx->emit_(OP_BITWISE_OR, BC_NOARG, line); break;
  591. case TK("^"): ctx->emit_(OP_BITWISE_XOR, BC_NOARG, line); break;
  592. case TK("@"): ctx->emit_(OP_BINARY_MATMUL, BC_NOARG, line); break;
  593. default: PK_FATAL_ERROR();
  594. }
  595. for(int i: jmps) ctx->patch_jump(i);
  596. }
  597. void TernaryExpr::emit_(CodeEmitContext* ctx){
  598. cond->emit_(ctx);
  599. int patch = ctx->emit_(OP_POP_JUMP_IF_FALSE, BC_NOARG, cond->line);
  600. true_expr->emit_(ctx);
  601. int patch_2 = ctx->emit_(OP_JUMP_ABSOLUTE, BC_NOARG, true_expr->line);
  602. ctx->patch_jump(patch);
  603. false_expr->emit_(ctx);
  604. ctx->patch_jump(patch_2);
  605. }
  606. } // namespace pkpy