summaryrefslogtreecommitdiffstats
path: root/sys/sys
diff options
context:
space:
mode:
authorpeter <peter@FreeBSD.org>2001-06-13 10:58:39 +0000
committerpeter <peter@FreeBSD.org>2001-06-13 10:58:39 +0000
commitf10fa038c14063eaf2da32ed734e644e5f569694 (patch)
tree88aef8097c80f09c2f725d61b6da4d433a595a61 /sys/sys
parent2514663dd721b9ac234d35a4dac65dfc457ff6bc (diff)
downloadFreeBSD-src-f10fa038c14063eaf2da32ed734e644e5f569694.zip
FreeBSD-src-f10fa038c14063eaf2da32ed734e644e5f569694.tar.gz
With this commit, I hereby pronounce gensetdefs past its use-by date.
Replace the a.out emulation of 'struct linker_set' with something a little more flexible. <sys/linker_set.h> now provides macros for accessing elements and completely hides the implementation. The linker_set.h macros have been on the back burner in various forms since 1998 and has ideas and code from Mike Smith (SET_FOREACH()), John Polstra (ELF clue) and myself (cleaned up API and the conversion of the rest of the kernel to use it). The macros declare a strongly typed set. They return elements with the type that you declare the set with, rather than a generic void *. For ELF, we use the magic ld symbols (__start_<setname> and __stop_<setname>). Thanks to Richard Henderson <rth@redhat.com> for the trick about how to force ld to provide them for kld's. For a.out, we use the old linker_set struct. NOTE: the item lists are no longer null terminated. This is why the code impact is high in certain areas. The runtime linker has a new method to find the linker set boundaries depending on which backend format is in use. linker sets are still module/kld unfriendly and should never be used for anything that may be modular one day. Reviewed by: eivind
Diffstat (limited to 'sys/sys')
-rw-r--r--sys/sys/cons.h1
-rw-r--r--sys/sys/kernel.h2
-rw-r--r--sys/sys/linker.h8
-rw-r--r--sys/sys/linker_set.h124
-rw-r--r--sys/sys/sysctl.h6
5 files changed, 98 insertions, 43 deletions
diff --git a/sys/sys/cons.h b/sys/sys/cons.h
index 48be371..a99c210 100644
--- a/sys/sys/cons.h
+++ b/sys/sys/cons.h
@@ -78,7 +78,6 @@ struct consdev {
#define CN_REMOTE 3 /* serial interface with remote bit set */
#ifdef _KERNEL
-extern struct linker_set cons_set;
extern int cons_unavail;
extern struct consdev *cn_tab;
diff --git a/sys/sys/kernel.h b/sys/sys/kernel.h
index 7e8ea63..79c57e1 100644
--- a/sys/sys/kernel.h
+++ b/sys/sys/kernel.h
@@ -250,7 +250,7 @@ struct sysinit {
C_SYSUNINIT(uniquifier, subsystem, order, \
(sysinit_cfunc_t)(sysinit_nfunc_t)func, (void *)ident)
-void sysinit_add __P((struct sysinit **set));
+void sysinit_add __P((struct sysinit **set, struct sysinit **set_end));
/*
* Infrastructure for tunable 'constants'. Value may be specified at compile
diff --git a/sys/sys/linker.h b/sys/sys/linker.h
index 87a2a83..671bbe8 100644
--- a/sys/sys/linker.h
+++ b/sys/sys/linker.h
@@ -139,6 +139,14 @@ caddr_t linker_file_lookup_symbol(linker_file_t _file, const char* _name,
int _deps);
/*
+ * Lookup a linker set in a file. Return pointers to the first entry,
+ * last + 1, and count of entries. Use: for (p = start; p < stop; p++) {}
+ * void *start is really: "struct yoursetmember ***start;"
+ */
+int linker_file_lookup_set(linker_file_t _file, const char *_name,
+ void *_start, void *_stop, int *_count);
+
+/*
* This routine is responsible for finding dependencies of userland
* initiated kldload(2)'s of files.
*/
diff --git a/sys/sys/linker_set.h b/sys/sys/linker_set.h
index 2bffa9b..0fa115f 100644
--- a/sys/sys/linker_set.h
+++ b/sys/sys/linker_set.h
@@ -1,5 +1,6 @@
/*-
* Copyright (c) 1999 John D. Polstra
+ * Copyright (c) 1999,2001 Peter Wemm <peter@FreeBSD.org>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -31,56 +32,109 @@
/*
* The following macros are used to declare global sets of objects, which
- * are collected by the linker into a `struct linker_set' as defined below.
+ * are collected by the linker into a `linker_set' as defined below.
* For ELF, this is done by constructing a separate segment for each set.
* For a.out, it is done automatically by the linker.
+ */
+
+#if defined(__ELF__)
+/*
+ * Private macros, not to be used outside this header file.
+ */
+/* this bit of h0h0magic brought to you by cpp */
+#define __GLOBL(sym) __GLOBL2(sym)
+#define __GLOBL2(sym) __asm(".globl " #sym)
+
+#define __MAKE_SET(set, sym) \
+ __GLOBL(__CONCAT(__start_set_,set)); \
+ __GLOBL(__CONCAT(__stop_set_,set)); \
+ static void const * const __set_##set##_sym_##sym \
+ __attribute__((__section__("set_" #set),__unused__)) = &sym
+
+/*
+ * Public macros.
+ */
+#define TEXT_SET(set, sym) __MAKE_SET(set, sym)
+#define DATA_SET(set, sym) __MAKE_SET(set, sym)
+#define BSS_SET(set, sym) __MAKE_SET(set, sym)
+#define ABS_SET(set, sym) __MAKE_SET(set, sym)
+#define SET_ENTRY(set, sym) __MAKE_SET(set, sym)
+
+/*
+ * Initialize before referring to a give linker set
+ */
+#define SET_DECLARE(set, ptype) \
+ extern ptype *__CONCAT(__start_set_,set); \
+ extern ptype *__CONCAT(__stop_set_,set)
+
+#define SET_BEGIN(set) \
+ (&__CONCAT(__start_set_,set))
+#define SET_LIMIT(set) \
+ (&__CONCAT(__stop_set_,set))
+
+#else /* __ELF__ */
+
+/*
+ * The old way. This depends on GNU ld extensions that are not widely
+ * available outside of the a.out format.
*
- * In the MAKE_SET macros below, the lines:
+ * NB: the constants defined below must match those defined in
+ * ld/ld.h. Since their calculation requires arithmetic, we
+ * can't name them symbolically (e.g., 23 is N_SETT | N_EXT).
*
+ * In the __MAKE_SET macro below, the line:
* static void const * const __set_##set##_sym_##sym = &sym;
- *
- * are present only to prevent the compiler from producing bogus
+ * is present only to prevent the compiler from producing bogus
* warnings about unused symbols.
*/
-#ifdef __ELF__
-
-#if defined(__alpha__) || defined(__ia64__)
-#define MAKE_SET(set, sym) \
+/* Private macros */
+#ifdef __UNDERSCORES__
+#define __MAKE_SET(set, sym, type) \
static void const * const __set_##set##_sym_##sym = &sym; \
- __asm(".section .set." #set ",\"aw\""); \
- __asm(".p2align 3"); \
- __asm(".quad " #sym); \
- __asm(".previous")
+ __asm(".stabs \"_" #set "\", " #type ", 0, 0, _" #sym)
#else
-#define MAKE_SET(set, sym) \
+#define __MAKE_SET(set, sym, type) \
static void const * const __set_##set##_sym_##sym = &sym; \
- __asm(".section .set." #set ",\"aw\""); \
- __asm(".p2align 2"); \
- __asm(".long " #sym); \
- __asm(".previous")
+ __asm(".stabs \"" #set "\", " #type ", 0, 0, " #sym)
#endif
-#define TEXT_SET(set, sym) MAKE_SET(set, sym)
-#define DATA_SET(set, sym) MAKE_SET(set, sym)
-#else
+/* Public Macros */
+#define TEXT_SET(set, sym) __MAKE_SET(set, sym, 23)
+#define DATA_SET(set, sym) __MAKE_SET(set, sym, 25)
+#define BSS_SET(set, sym) __MAKE_SET(set, sym, 27)
+#define ABS_SET(set, sym) __MAKE_SET(set, sym, 21)
+#define SET_ENTRY(set, sym) error error must provide text/data type
+
+#define SET_DECLARE(set, ptype) \
+ extern struct { \
+ int ls_length; \
+ ptype *ls_items[1]; \
+ } set
+
+#define SET_BEGIN(set) \
+ (&((set).ls_items[0]))
+#define SET_LIMIT(set) \
+ (&((set).ls_items[(set).ls_length]))
+
+#endif /* __ELF__ */
/*
- * NB: the constants defined below must match those defined in
- * ld/ld.h. Since their calculation requires arithmetic, we
- * can't name them symbolically (e.g., 23 is N_SETT | N_EXT).
+ * Iterate over all the elements of a set.
+ *
+ * Sets always contain addresses of things, and "pvar" points to words
+ * containing those addresses. Thus is must be declared as "type **pvar",
+ * and the address of each set item is obtained inside the loop by "*pvar".
*/
-#define MAKE_SET(set, sym, type) \
- static void const * const __set_##set##_sym_##sym = &sym; \
- __asm(".stabs \"" #set "\", " #type ", 0, 0, " #sym)
-#define TEXT_SET(set, sym) MAKE_SET(set, sym, 23)
-#define DATA_SET(set, sym) MAKE_SET(set, sym, 25)
+#define SET_FOREACH(pvar, set) \
+ for (pvar = SET_BEGIN(set); pvar < SET_LIMIT(set); pvar++)
-#endif
+#define SET_ITEM(set, i) \
+ ((SET_BEGIN(set))[i])
-struct linker_set {
- int ls_length;
- void *ls_items[1]; /* really ls_length of them,
- * trailing NULL */
-};
+/*
+ * Provide a count of the items in a set.
+ */
+#define SET_COUNT(set) \
+ (SET_LIMIT(set) - SET_BEGIN(set))
-#endif /* _SYS_LINKER_SET_H_ */
+#endif /* _SYS_LINKER_SET_H_ */
diff --git a/sys/sys/sysctl.h b/sys/sys/sysctl.h
index d43c89f8..1b6657b 100644
--- a/sys/sys/sysctl.h
+++ b/sys/sys/sysctl.h
@@ -557,8 +557,6 @@ extern char machine[];
extern char osrelease[];
extern char ostype[];
-struct linker_set;
-
/* Dynamic oid handling */
struct sysctl_oid *sysctl_add_oid(struct sysctl_ctx_list *clist,
struct sysctl_oid_list *parent, int nbr, const char *name,
@@ -575,10 +573,6 @@ struct sysctl_ctx_entry *sysctl_ctx_entry_find(struct sysctl_ctx_list *clist,
int sysctl_ctx_entry_del(struct sysctl_ctx_list *clist,
struct sysctl_oid *oidp);
-/* Linker set based oid handling */
-void sysctl_register_set(struct linker_set *lsp);
-void sysctl_unregister_set(struct linker_set *lsp);
-
int kernel_sysctl(struct proc *p, int *name, u_int namelen, void *old,
size_t *oldlenp, void *new, size_t newlen,
size_t *retval);
OpenPOWER on IntegriCloud