aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/lj_cconv.c4
-rw-r--r--src/lj_crecord.c49
2 files changed, 50 insertions, 3 deletions
diff --git a/src/lj_cconv.c b/src/lj_cconv.c
index ab398adc..13b8230d 100644
--- a/src/lj_cconv.c
+++ b/src/lj_cconv.c
@@ -448,8 +448,10 @@ int lj_cconv_tv_bf(CTState *cts, CType *s, TValue *o, uint8_t *sp)
448 setintV(o, (int32_t)val); 448 setintV(o, (int32_t)val);
449 } 449 }
450 } else { 450 } else {
451 uint32_t b = (val >> pos) & 1;
451 lua_assert(bsz == 1); 452 lua_assert(bsz == 1);
452 setboolV(o, (val >> pos) & 1); 453 setboolV(o, b);
454 setboolV(&cts->g->tmptv2, b); /* Remember for trace recorder. */
453 } 455 }
454 return 0; /* No GC step needed. */ 456 return 0; /* No GC step needed. */
455} 457}
diff --git a/src/lj_crecord.c b/src/lj_crecord.c
index 2db4c86b..e32ae23e 100644
--- a/src/lj_crecord.c
+++ b/src/lj_crecord.c
@@ -751,6 +751,48 @@ static void crec_index_meta(jit_State *J, CTState *cts, CType *ct,
751 } 751 }
752} 752}
753 753
754/* Record bitfield load/store. */
755static void crec_index_bf(jit_State *J, RecordFFData *rd, TRef ptr, CTInfo info)
756{
757 IRType t = IRT_I8 + 2*lj_fls(ctype_bitcsz(info)) + ((info&CTF_UNSIGNED)?1:0);
758 TRef tr = emitir(IRT(IR_XLOAD, t), ptr, 0);
759 CTSize pos = ctype_bitpos(info), bsz = ctype_bitbsz(info), shift = 32 - bsz;
760 lua_assert(t <= IRT_U32); /* NYI: 64 bit bitfields. */
761 if (rd->data == 0) { /* __index metamethod. */
762 if ((info & CTF_BOOL)) {
763 tr = emitir(IRTI(IR_BAND), tr, lj_ir_kint(J, (int32_t)((1u << pos))));
764 /* Assume not equal to zero. Fixup and emit pending guard later. */
765 lj_ir_set(J, IRTGI(IR_NE), tr, lj_ir_kint(J, 0));
766 J->postproc = LJ_POST_FIXGUARD;
767 tr = TREF_TRUE;
768 } else if (!(info & CTF_UNSIGNED)) {
769 tr = emitir(IRTI(IR_BSHL), tr, lj_ir_kint(J, shift - pos));
770 tr = emitir(IRTI(IR_BSAR), tr, lj_ir_kint(J, shift));
771 } else {
772 lua_assert(bsz < 32); /* Full-size fields cannot end up here. */
773 tr = emitir(IRTI(IR_BSHR), tr, lj_ir_kint(J, pos));
774 tr = emitir(IRTI(IR_BAND), tr, lj_ir_kint(J, (int32_t)((1u << bsz)-1)));
775 /* We can omit the U32 to NUM conversion, since bsz < 32. */
776 }
777 J->base[0] = tr;
778 } else { /* __newindex metamethod. */
779 CTState *cts = ctype_ctsG(J2G(J));
780 CType *ct = ctype_get(cts,
781 (info & CTF_BOOL) ? CTID_BOOL :
782 (info & CTF_UNSIGNED) ? CTID_UINT32 : CTID_INT32);
783 int32_t mask = (int32_t)(((1u << bsz)-1) << pos);
784 TRef sp = crec_ct_tv(J, ct, 0, J->base[2], &rd->argv[2]);
785 sp = emitir(IRTI(IR_BSHL), sp, lj_ir_kint(J, pos));
786 /* Use of the target type avoids forwarding conversions. */
787 sp = emitir(IRT(IR_BAND, t), sp, lj_ir_kint(J, mask));
788 tr = emitir(IRT(IR_BAND, t), tr, lj_ir_kint(J, (int32_t)~mask));
789 tr = emitir(IRT(IR_BOR, t), tr, sp);
790 emitir(IRT(IR_XSTORE, t), ptr, tr);
791 rd->nres = 0;
792 J->needsnap = 1;
793 }
794}
795
754void LJ_FASTCALL recff_cdata_index(jit_State *J, RecordFFData *rd) 796void LJ_FASTCALL recff_cdata_index(jit_State *J, RecordFFData *rd)
755{ 797{
756 TRef idx, ptr = J->base[0]; 798 TRef idx, ptr = J->base[0];
@@ -825,6 +867,7 @@ again:
825 CType *fct; 867 CType *fct;
826 fct = lj_ctype_getfield(cts, ct, name, &fofs); 868 fct = lj_ctype_getfield(cts, ct, name, &fofs);
827 if (fct) { 869 if (fct) {
870 ofs += (ptrdiff_t)fofs;
828 /* Always specialize to the field name. */ 871 /* Always specialize to the field name. */
829 emitir(IRTG(IR_EQ, IRT_STR), idx, lj_ir_kstr(J, name)); 872 emitir(IRTG(IR_EQ, IRT_STR), idx, lj_ir_kstr(J, name));
830 if (ctype_isconstval(fct->info)) { 873 if (ctype_isconstval(fct->info)) {
@@ -836,12 +879,14 @@ again:
836 J->base[0] = lj_ir_kint(J, (int32_t)fct->size); 879 J->base[0] = lj_ir_kint(J, (int32_t)fct->size);
837 return; /* Interpreter will throw for newindex. */ 880 return; /* Interpreter will throw for newindex. */
838 } else if (ctype_isbitfield(fct->info)) { 881 } else if (ctype_isbitfield(fct->info)) {
839 lj_trace_err(J, LJ_TRERR_NYICONV); 882 if (ofs)
883 ptr = emitir(IRT(IR_ADD, IRT_PTR), ptr, lj_ir_kintp(J, ofs));
884 crec_index_bf(J, rd, ptr, fct->info);
885 return;
840 } else { 886 } else {
841 lua_assert(ctype_isfield(fct->info)); 887 lua_assert(ctype_isfield(fct->info));
842 sid = ctype_cid(fct->info); 888 sid = ctype_cid(fct->info);
843 } 889 }
844 ofs += (ptrdiff_t)fofs;
845 } 890 }
846 } else if (ctype_iscomplex(ct->info)) { 891 } else if (ctype_iscomplex(ct->info)) {
847 if (name->len == 2 && 892 if (name->len == 2 &&