// SPDX-License-Identifier: GPL-2.0 #ifndef pr_fmt #define pr_fmt(fmt) "alt: " fmt #endif #include #include #include #include #include #include #include #include #ifndef a_debug #define a_debug pr_debug #endif #ifndef __kernel_va #define __kernel_va(x) (void *)(x) #endif unsigned long __bootdata_preserved(machine_features[1]); struct alt_debug { unsigned long facilities[MAX_FACILITY_BIT / BITS_PER_LONG]; unsigned long mfeatures[MAX_MFEATURE_BIT / BITS_PER_LONG]; int spec; }; static struct alt_debug __bootdata_preserved(alt_debug); static void alternative_dump(u8 *old, u8 *new, unsigned int len, unsigned int type, unsigned int data) { char oinsn[33], ninsn[33]; unsigned long kptr; unsigned int pos; for (pos = 0; pos < len && 2 * pos < sizeof(oinsn) - 3; pos++) hex_byte_pack(&oinsn[2 * pos], old[pos]); oinsn[2 * pos] = 0; for (pos = 0; pos < len && 2 * pos < sizeof(ninsn) - 3; pos++) hex_byte_pack(&ninsn[2 * pos], new[pos]); ninsn[2 * pos] = 0; kptr = (unsigned long)__kernel_va(old); a_debug("[%d/%3d] %016lx: %s -> %s\n", type, data, kptr, oinsn, ninsn); } void __apply_alternatives(struct alt_instr *start, struct alt_instr *end, unsigned int ctx) { struct alt_debug *d; struct alt_instr *a; bool debug, replace; u8 *old, *new; /* * The scan order should be from start to end. A later scanned * alternative code can overwrite previously scanned alternative code. */ d = &alt_debug; for (a = start; a < end; a++) { if (!(a->ctx & ctx)) continue; switch (a->type) { case ALT_TYPE_FACILITY: replace = test_facility(a->data); debug = __test_facility(a->data, d->facilities); break; case ALT_TYPE_FEATURE: replace = test_machine_feature(a->data); debug = __test_machine_feature(a->data, d->mfeatures); break; case ALT_TYPE_SPEC: replace = nobp_enabled(); debug = d->spec; break; default: replace = false; debug = false; } if (!replace) continue; old = (u8 *)&a->instr_offset + a->instr_offset; new = (u8 *)&a->repl_offset + a->repl_offset; if (debug) alternative_dump(old, new, a->instrlen, a->type, a->data); s390_kernel_write(old, new, a->instrlen); } }