Write an unsigned 8-bit value to memory via an effective address.
(wme08 proc-mode eff-addr seg-reg val x86) → (mv flg x86-new)
The effective address
Prior to the effective address translation, we check whether write access is allowed. In 32-bit mode, write access is allowed in data segments (DS, ES, FS, GS, and SS) if the W bit in the segment descriptor is 1; write access is disallowed in code segments (this is not explicitly mentioned in Intel manual, May'18, Volume 3, Section 3.4.5.1, but it seems reasonable). In 64-bit mode, the W bit is ignored (see Atmel manual, Dec'17, Volume 2, Section 4.8.1); by analogy, we allow write access to the code segment as well. These checks may be slightly optimized using the invariant that SS.W must be 1 when SS is loaded.
Function:
(defun wme08$inline (proc-mode eff-addr seg-reg val x86) (declare (xargs :stobjs (x86))) (declare (type (integer 0 4) proc-mode) (type (signed-byte 64) eff-addr) (type (integer 0 5) seg-reg) (type (unsigned-byte 8) val)) (b* (((when (and (/= proc-mode 0) (or (= seg-reg 1) (b* ((attr (loghead 16 (seg-hidden-attri seg-reg x86))) (w (data-segment-descriptor-attributesbits->w attr))) (= w 0))))) (mv (list :non-writable-segment eff-addr seg-reg) x86)) ((mv flg lin-addr) (ea-to-la proc-mode eff-addr seg-reg 1 x86)) ((when flg) (mv flg x86))) (wml08 lin-addr val x86)))
Theorem:
(defthm x86p-of-wme08.x86-new (implies (x86p x86) (b* (((mv ?flg ?x86-new) (wme08$inline proc-mode eff-addr seg-reg val x86))) (x86p x86-new))) :rule-classes :rewrite)
Theorem:
(defthm wme08-when-64-bit-modep-and-not-fs/gs (implies (and (not (equal seg-reg 4)) (not (equal seg-reg 5)) (canonical-address-p eff-addr)) (equal (wme08 0 eff-addr seg-reg val x86) (wml08 eff-addr val x86))))
Theorem:
(defthm wme08-when-64-bit-modep-and-fs/gs (implies (or (equal seg-reg 4) (equal seg-reg 5)) (equal (wme08 0 eff-addr seg-reg val x86) (b* (((mv flg lin-addr) (b* (((mv base & &) (segment-base-and-bounds 0 seg-reg x86)) (lin-addr (i64 (+ base (n64 eff-addr))))) (if (canonical-address-p lin-addr) (mv nil lin-addr) (mv (list :non-canonical-address lin-addr) 0)))) ((when flg) (mv flg x86))) (wml08 lin-addr val x86)))))