Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src Work in progress for MIPS modules. Only tested on mipseb64,...
details: https://anonhg.NetBSD.org/src/rev/f4a143c3093b
branches: trunk
changeset: 981848:f4a143c3093b
user: simonb <simonb%NetBSD.org@localhost>
date: Tue Mar 23 13:22:40 2021 +0000
description:
Work in progress for MIPS modules. Only tested on mipseb64, not yet
enabled anywhere.
diffstat:
share/mk/bsd.kmodule.mk | 8 +-
sys/arch/mips/conf/files.mips | 3 +-
sys/arch/mips/mips/kobj_machdep.c | 257 ++++++++++++++++++++++++++++++++++++++
sys/arch/mips/mips/mips_machdep.c | 17 ++-
4 files changed, 281 insertions(+), 4 deletions(-)
diffs (truncated from 345 to 300 lines):
diff -r 7ab7ec08dacf -r f4a143c3093b share/mk/bsd.kmodule.mk
--- a/share/mk/bsd.kmodule.mk Tue Mar 23 13:19:09 2021 +0000
+++ b/share/mk/bsd.kmodule.mk Tue Mar 23 13:22:40 2021 +0000
@@ -1,4 +1,4 @@
-# $NetBSD: bsd.kmodule.mk,v 1.74 2021/03/07 07:37:35 rin Exp $
+# $NetBSD: bsd.kmodule.mk,v 1.75 2021/03/23 13:22:40 simonb Exp $
# We are not building this with PIE
MKPIE=no
@@ -63,6 +63,12 @@
LDFLAGS+= -Wl,-m,elf64ltsmip
.endif
+.if ${MACHINE_CPU} == "mips"
+# We can't use -msym32 with -mlong-calls as -msym32 forces all addresses
+# to be 32-bit which defeats the whole purpose of long calls.
+CFLAGS+= -mlong-calls
+.endif
+
.if ${MACHINE_CPU} == "sparc64"
# force same memory model as rest of the kernel
CFLAGS+= ${${ACTIVE_CC} == "gcc":? -mcmodel=medlow :}
diff -r 7ab7ec08dacf -r f4a143c3093b sys/arch/mips/conf/files.mips
--- a/sys/arch/mips/conf/files.mips Tue Mar 23 13:19:09 2021 +0000
+++ b/sys/arch/mips/conf/files.mips Tue Mar 23 13:22:40 2021 +0000
@@ -1,4 +1,4 @@
-# $NetBSD: files.mips,v 1.82 2021/03/23 11:41:53 simonb Exp $
+# $NetBSD: files.mips,v 1.83 2021/03/23 13:22:40 simonb Exp $
#
defflag opt_cputype.h NOFPU FPEMUL
@@ -45,6 +45,7 @@
file arch/mips/mips/db_trace.c ddb
file arch/mips/mips/ipifuncs.c multiprocessor
file arch/mips/mips/kgdb_machdep.c kgdb
+file arch/mips/mips/kobj_machdep.c modular
file arch/mips/mips/pmap_machdep.c
file arch/mips/mips/mips_stacktrace.c
file uvm/pmap/pmap.c
diff -r 7ab7ec08dacf -r f4a143c3093b sys/arch/mips/mips/kobj_machdep.c
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/sys/arch/mips/mips/kobj_machdep.c Tue Mar 23 13:22:40 2021 +0000
@@ -0,0 +1,257 @@
+/* $NetBSD: kobj_machdep.c,v 1.1 2021/03/23 13:22:40 simonb Exp $ */
+
+/*-
+ * Copyright (c) 2021 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Simon Burge.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
+ * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
+ * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
+ * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <sys/cdefs.h>
+
+__RCSID("$NetBSD");
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/exec_elf.h>
+#include <sys/kernel.h>
+#include <sys/kobj.h>
+#include <sys/xcall.h>
+
+#include <mips/cache.h>
+
+#ifndef __mips_n64
+#error update for non-N64 abis /* XXX */
+#endif
+
+#define RELOC_LO16(x) ((x) & 0xffff)
+#define RELOC_HI16(x) ((((int64_t)(x) + 0x8000LL) >> 16) & 0xffff)
+#define RELOC_HIGHER(x) ((((int64_t)(x) + 0x80008000LL) >> 32) & 0xffff)
+#define RELOC_HIGHEST(x) ((((int64_t)(x) + 0x800080008000LL) >> 48) & 0xffff)
+
+#undef MIPS_MODULE_DEBUG
+#ifdef MIPS_MODULE_DEBUG
+#define DPRINTF(fmt, args...) \
+ do { printf(fmt, ## args); } while (0)
+#else
+#define DPRINTF(fmt, args...) __nothing
+#endif
+
+int
+kobj_reloc(kobj_t ko, uintptr_t relocbase, const void *data,
+ bool isrela, bool local)
+{
+ Elf_Addr addr, addend, *where;
+ Elf32_Addr *where32;
+ Elf_Word rtype, symidx;
+ const Elf_Rela *rela;
+ uint32_t *insn;
+ int error;
+
+ DPRINTF("%s(kobj %p, reloc %#lx,\n data %p, rela %d, local %d)\n",
+ __func__, ko, relocbase, data, isrela, local);
+
+ if (!isrela) {
+ printf("%s: REL relocations not supported", __func__);
+ return -1;
+ }
+
+ rela = (const Elf_Rela *)data;
+ where = (Elf_Addr *)(relocbase + rela->r_offset);
+ addend = rela->r_addend;
+ rtype = ELF_R_TYPE(rela->r_info);
+ symidx = ELF_R_SYM(rela->r_info);
+
+ /* pointer to 32bit value and instruction */
+ insn = (void *)where;
+
+ /* check alignment */
+ KASSERT(((intptr_t)where & (sizeof(int32_t) - 1)) == 0);
+
+ switch (rtype) {
+ case R_TYPE(NONE): /* none */
+ DPRINTF(" reloc R_MIPS_NONE\n");
+ break;
+ /* R_TYPE(16) */
+ case R_TYPE(32): /* S + A */
+ DPRINTF(" reloc R_MIPS_32\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+ addr += addend;
+ where32 = (void *)where;
+ DPRINTF(" orig = 0x%08x\n", *where32);
+ *where32 = addr;
+ DPRINTF(" new = 0x%08x\n", *where32);
+ break;
+ /* R_TYPE(REL32) */
+ case R_TYPE(26): /* (((A << 2) | (P & 0xf0000000)) + S) >> 2 */
+ /* XXXXXX untested */
+ DPRINTF(" reloc R_MIPS_26 (untested)\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+
+ addend &= __BITS(25, 0); /* mask off lower 26 bits */
+ addend <<= 2;
+
+ addr += ((intptr_t)where & 0xf0000000) | addend;
+ addr >>= 2;
+
+ KASSERT((*insn & 0x3ffffff) == 0);
+ DPRINTF(" orig insn = 0x%08x\n", *insn);
+ *insn |= addr;
+ DPRINTF(" new insn = 0x%08x\n", *insn);
+
+ break;
+ case R_TYPE(HI16): /* %high(AHL + S) = (x - (short)x) >> 16 */
+ DPRINTF(" reloc R_MIPS_HI16\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+
+ addr += addend;
+ KASSERT((*insn & 0xffff) == 0);
+ DPRINTF(" orig insn = 0x%08x\n", *insn);
+ DPRINTF(" HI16(%#lx) = 0x%04llx\n", addr, RELOC_HI16(addr));
+ *insn |= RELOC_HI16(addr);
+ DPRINTF(" new insn = 0x%08x\n", *insn);
+ break;
+ case R_TYPE(LO16): /* AHL + S */
+ DPRINTF(" reloc R_MIPS_LO16\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+
+ addr += addend;
+ KASSERT((*insn & 0xffff) == 0);
+ DPRINTF(" orig insn = 0x%08x\n", *insn);
+ DPRINTF(" LO16(%#lx) = 0x%04lx\n", addr, RELOC_LO16(addr));
+ *insn |= RELOC_LO16(addr);
+ DPRINTF(" new insn = 0x%08x\n", *insn);
+ break;
+ /* R_TYPE(GPREL16) */
+ /* R_TYPE(LITERAL) */
+ /* R_TYPE(GOT16) */
+ /* R_TYPE(PC16) */
+ /* R_TYPE(CALL16) */
+#ifdef _LP64
+ /* R_TYPE(GPREL32) */
+ /* R_TYPE(SHIFT5) */
+ /* R_TYPE(SHIFT6) */
+ case R_TYPE(64): /* S + A */
+ DPRINTF(" reloc R_MIPS_64\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+
+ addr += addend;
+ DPRINTF(" orig = 0x%016lx\n", *where);
+ *where = addr;
+ DPRINTF(" new = 0x%016lx\n", *where);
+ break;
+ /* R_TYPE(GOT_DISP) */
+ /* R_TYPE(GOT_PAGE) */
+ /* R_TYPE(GOT_OFST) */
+ /* R_TYPE(GOT_HI16) */
+ /* R_TYPE(GOT_LO16) */
+ /* R_TYPE(SUB) */
+ /* R_TYPE(INSERT_A) */
+ /* R_TYPE(INSERT_B) */
+ /* R_TYPE(DELETE) */
+ case R_TYPE(HIGHER): /* %higher(A + S) =
+ (((long long)x + 0x80008000LL) >> 32) & 0xffff */
+ DPRINTF(" reloc R_MIPS_HIGHER\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+
+ addr += addend;
+ KASSERT((*insn & 0xffff) == 0);
+ DPRINTF(" orig insn = 0x%08x\n", *insn);
+ DPRINTF(" HIGHER(%#lx) = 0x%04llx\n", addr, RELOC_HIGHER(addr));
+ *insn |= RELOC_HIGHER(addr);
+ DPRINTF(" new insn = 0x%08x\n", *insn);
+ break;
+ case R_TYPE(HIGHEST): /* %highest(A + S) =
+ (((long long)x + 0x800080008000LL) >> 48) & 0xffff */
+ DPRINTF(" reloc R_MIPS_HIGHEST\n");
+ error = kobj_sym_lookup(ko, symidx, &addr);
+ if (error)
+ return -1;
+
+ addr += addend;
+ KASSERT((*insn & 0xffff) == 0);
+ DPRINTF(" orig insn = 0x%08x\n", *insn);
+ DPRINTF(" HIGHEST(%#lx) = 0x%04llx\n", addr, RELOC_HIGHEST(addr));
+ *insn |= RELOC_HIGHEST(addr);
+ DPRINTF(" new insn = 0x%08x\n", *insn);
+ break;
+ /* R_TYPE(CALL_HI16) */
+ /* R_TYPE(CALL_LO16) */
+ /* R_TYPE(SCN_DISP) */
+ /* R_TYPE(REL16) */
+ /* R_TYPE(ADD_IMMEDIATE) */
+ /* R_TYPE(PJUMP) */
+ /* R_TYPE(RELGOT) */
+ /* R_TYPE(JALR) */
+#endif /* _LP64 */
+ default:
+ printf("%s: unknown reloc type %d @ %p\n",
+ __func__, rtype, where);
+ return -1;
+ }
+
+ return 0;
+}
+
+static void
+kobj_idcache_wbinv_all(void)
+{
+
+ mips_icache_sync_all();
+ mips_dcache_wbinv_all(); /* XXX needed? */
+}
+
+int
+kobj_machdep(kobj_t ko, void *base, size_t size, bool load)
+{
+
+ uint64_t where;
+
+ DPRINTF("%s(kobj %p, base %p,\n size %zu, load %d)\n",
+ __func__, ko, base, size, load);
+ if (load) {
+ if (cold) {
+ kobj_idcache_wbinv_all();
+ } else {
+ where = xc_broadcast(0,
+ (xcfunc_t)kobj_idcache_wbinv_all, NULL, NULL);
+ xc_wait(where);
+ }
+ }
+
+ return 0;
+}
Home |
Main Index |
Thread Index |
Old Index