diff options
-rw-r--r-- | src/lj_cconv.c | 4 | ||||
-rw-r--r-- | src/lj_crecord.c | 49 |
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. */ | ||
755 | static 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 | |||
754 | void LJ_FASTCALL recff_cdata_index(jit_State *J, RecordFFData *rd) | 796 | void 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 && |