Module Name:    src
Committed By:   joerg
Date:           Sat Mar 12 22:54:37 UTC 2011

Modified Files:
        src/libexec/ld.elf_so: rtld.h symbol.c
        src/libexec/ld.elf_so/arch/i386: mdreloc.c
        src/libexec/ld.elf_so/arch/sh3: mdreloc.c
        src/libexec/ld.elf_so/arch/x86_64: mdreloc.c
        src/sys/arch/amd64/include: types.h
        src/sys/arch/i386/include: types.h
        src/sys/arch/sh3/include: types.h

Log Message:
Add TLS support for AMD64, i386 and SH3.

This material is based upon work partially supported by
The NetBSD Foundation under a contract with Joerg Sonnenberger.


To generate a diff of this commit:
cvs rdiff -u -r1.102 -r1.103 src/libexec/ld.elf_so/rtld.h
cvs rdiff -u -r1.55 -r1.56 src/libexec/ld.elf_so/symbol.c
cvs rdiff -u -r1.32 -r1.33 src/libexec/ld.elf_so/arch/i386/mdreloc.c
cvs rdiff -u -r1.28 -r1.29 src/libexec/ld.elf_so/arch/sh3/mdreloc.c
cvs rdiff -u -r1.38 -r1.39 src/libexec/ld.elf_so/arch/x86_64/mdreloc.c
cvs rdiff -u -r1.36 -r1.37 src/sys/arch/amd64/include/types.h
cvs rdiff -u -r1.71 -r1.72 src/sys/arch/i386/include/types.h
cvs rdiff -u -r1.31 -r1.32 src/sys/arch/sh3/include/types.h

Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.

Modified files:

Index: src/libexec/ld.elf_so/rtld.h
diff -u src/libexec/ld.elf_so/rtld.h:1.102 src/libexec/ld.elf_so/rtld.h:1.103
--- src/libexec/ld.elf_so/rtld.h:1.102	Thu Mar 10 14:27:31 2011
+++ src/libexec/ld.elf_so/rtld.h	Sat Mar 12 22:54:36 2011
@@ -1,4 +1,4 @@
-/*	$NetBSD: rtld.h,v 1.102 2011/03/10 14:27:31 joerg Exp $	 */
+/*	$NetBSD: rtld.h,v 1.103 2011/03/12 22:54:36 joerg Exp $	 */
 
 /*
  * Copyright 1996 John D. Polstra.
@@ -375,6 +375,10 @@
 extern size_t _rtld_tls_max_index;
 
 __dso_public extern void *__tls_get_addr(void *);
+#ifdef __i386__
+__dso_public extern void *___tls_get_addr(void *)
+    __attribute__((__regparm__(1)));
+#endif
 #endif
 
 /* map_object.c */

Index: src/libexec/ld.elf_so/symbol.c
diff -u src/libexec/ld.elf_so/symbol.c:1.55 src/libexec/ld.elf_so/symbol.c:1.56
--- src/libexec/ld.elf_so/symbol.c:1.55	Wed Mar  9 23:10:07 2011
+++ src/libexec/ld.elf_so/symbol.c	Sat Mar 12 22:54:36 2011
@@ -1,4 +1,4 @@
-/*	$NetBSD: symbol.c,v 1.55 2011/03/09 23:10:07 joerg Exp $	 */
+/*	$NetBSD: symbol.c,v 1.56 2011/03/12 22:54:36 joerg Exp $	 */
 
 /*
  * Copyright 1996 John D. Polstra.
@@ -40,7 +40,7 @@
 
 #include <sys/cdefs.h>
 #ifndef lint
-__RCSID("$NetBSD: symbol.c,v 1.55 2011/03/09 23:10:07 joerg Exp $");
+__RCSID("$NetBSD: symbol.c,v 1.56 2011/03/12 22:54:36 joerg Exp $");
 #endif /* not lint */
 
 #include <err.h>
@@ -97,6 +97,9 @@
 		(fptr_t)_rtld_tls_allocate,
 		(fptr_t)_rtld_tls_free,
 		(fptr_t)__tls_get_addr,
+#ifdef __i386__
+		(fptr_t)___tls_get_addr,
+#endif
 #endif
 		NULL
 	};

Index: src/libexec/ld.elf_so/arch/i386/mdreloc.c
diff -u src/libexec/ld.elf_so/arch/i386/mdreloc.c:1.32 src/libexec/ld.elf_so/arch/i386/mdreloc.c:1.33
--- src/libexec/ld.elf_so/arch/i386/mdreloc.c:1.32	Fri Aug  6 16:33:18 2010
+++ src/libexec/ld.elf_so/arch/i386/mdreloc.c	Sat Mar 12 22:54:36 2011
@@ -1,11 +1,12 @@
-/*	$NetBSD: mdreloc.c,v 1.32 2010/08/06 16:33:18 joerg Exp $	*/
+/*	$NetBSD: mdreloc.c,v 1.33 2011/03/12 22:54:36 joerg Exp $	*/
 
 #include <sys/cdefs.h>
 #ifndef lint
-__RCSID("$NetBSD: mdreloc.c,v 1.32 2010/08/06 16:33:18 joerg Exp $");
+__RCSID("$NetBSD: mdreloc.c,v 1.33 2011/03/12 22:54:36 joerg Exp $");
 #endif /* not lint */
 
 #include <sys/types.h>
+#include <sys/ucontext.h>
 
 #include "debug.h"
 #include "rtld.h"
@@ -119,6 +120,47 @@
 			rdbg(("COPY (avoid in main)"));
 			break;
 
+		case R_TYPE(TLS_TPOFF):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			if (!defobj->tls_done &&
+			    _rtld_tls_offset_allocate(obj))
+				return -1;
+
+			*where = (Elf_Addr)(def->st_value - defobj->tlsoffset);
+
+			rdbg(("TLS_TPOFF %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where));
+			break;
+
+		case R_TYPE(TLS_DTPMOD32):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			*where = (Elf_Addr)(defobj->tlsindex);
+
+			rdbg(("TLS_DTPMOD32 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where));
+			break;
+
+		case R_TYPE(TLS_DTPOFF32):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			*where = (Elf_Addr)(def->st_value);
+
+			rdbg(("TLS_DTPOFF32 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where));
+
+			break;
+
 		default:
 			rdbg(("sym = %lu, type = %lu, offset = %p, "
 			    "contents = %p, symbol = %s",
@@ -214,3 +256,25 @@
 	}
 	return err;
 }
+
+/*
+ * i386 specific GNU variant of __tls_get_addr using register based
+ * argument passing.
+ */
+#define	DTV_MAX_INDEX(dtv)	((size_t)((dtv)[-1]))
+
+__dso_public __attribute__((__regparm__(1))) void *
+___tls_get_addr(void *arg_)
+{
+	size_t *arg = (size_t *)arg_;
+	void **dtv;
+	struct tls_tcb *tcb = __lwp_getprivate_fast();
+	size_t idx = arg[0], offset = arg[1];
+
+	dtv = tcb->tcb_dtv;
+
+	if (__predict_true(idx < DTV_MAX_INDEX(dtv) && dtv[idx] != NULL))
+		return (uint8_t *)dtv[idx] + offset;
+
+	return _rtld_tls_get_addr(tcb, idx, offset);
+}

Index: src/libexec/ld.elf_so/arch/sh3/mdreloc.c
diff -u src/libexec/ld.elf_so/arch/sh3/mdreloc.c:1.28 src/libexec/ld.elf_so/arch/sh3/mdreloc.c:1.29
--- src/libexec/ld.elf_so/arch/sh3/mdreloc.c:1.28	Fri Aug  6 16:33:18 2010
+++ src/libexec/ld.elf_so/arch/sh3/mdreloc.c	Sat Mar 12 22:54:36 2011
@@ -1,16 +1,17 @@
-/*	$NetBSD: mdreloc.c,v 1.28 2010/08/06 16:33:18 joerg Exp $	*/
+/*	$NetBSD: mdreloc.c,v 1.29 2011/03/12 22:54:36 joerg Exp $	*/
 
 #include <sys/cdefs.h>
 #ifndef lint
-__RCSID("$NetBSD: mdreloc.c,v 1.28 2010/08/06 16:33:18 joerg Exp $");
+__RCSID("$NetBSD: mdreloc.c,v 1.29 2011/03/12 22:54:36 joerg Exp $");
 #endif /* not lint */
 
 #include <sys/cdefs.h>
 #ifndef lint
-__RCSID("$NetBSD: mdreloc.c,v 1.28 2010/08/06 16:33:18 joerg Exp $");
+__RCSID("$NetBSD: mdreloc.c,v 1.29 2011/03/12 22:54:36 joerg Exp $");
 #endif /* not lint */
 
 #include <sys/types.h>
+#include <sys/tls.h>
 
 #include "debug.h"
 #include "rtld.h"
@@ -154,6 +155,49 @@
 			rdbg(("COPY (avoid in main)"));
 			break;
 
+		case R_TYPE(TLS_DTPOFF32):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			*where = (Elf_Addr)(def->st_value);
+
+			rdbg(("TLS_DTPOFF32 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where));
+
+			break;
+		case R_TYPE(TLS_DTPMOD32):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			*where = (Elf_Addr)(defobj->tlsindex);
+
+			rdbg(("TLS_DTPMOD32 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where));
+
+			break;
+
+		case R_TYPE(TLS_TPOFF32):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			if (!defobj->tls_done &&
+			    _rtld_tls_offset_allocate(obj))
+				return -1;
+
+			*where = (Elf_Addr)def->st_value +
+			    rela->r_addend + defobj->tlsoffset +
+			    sizeof(struct tls_tcb);
+
+			rdbg(("TLS_TPOFF32 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where));
+			break;
+
 		default:
 			rdbg(("sym = %lu, type = %lu, offset = %p, "
 			    "addend = %p, contents = %p, symbol = %s",

Index: src/libexec/ld.elf_so/arch/x86_64/mdreloc.c
diff -u src/libexec/ld.elf_so/arch/x86_64/mdreloc.c:1.38 src/libexec/ld.elf_so/arch/x86_64/mdreloc.c:1.39
--- src/libexec/ld.elf_so/arch/x86_64/mdreloc.c:1.38	Fri Aug  6 16:33:19 2010
+++ src/libexec/ld.elf_so/arch/x86_64/mdreloc.c	Sat Mar 12 22:54:36 2011
@@ -1,4 +1,4 @@
-/*	$NetBSD: mdreloc.c,v 1.38 2010/08/06 16:33:19 joerg Exp $	*/
+/*	$NetBSD: mdreloc.c,v 1.39 2011/03/12 22:54:36 joerg Exp $	*/
 
 /*
  * Copyright (c) 2001 Wasabi Systems, Inc.
@@ -68,7 +68,7 @@
 
 #include <sys/cdefs.h>
 #ifndef lint
-__RCSID("$NetBSD: mdreloc.c,v 1.38 2010/08/06 16:33:19 joerg Exp $");
+__RCSID("$NetBSD: mdreloc.c,v 1.39 2011/03/12 22:54:36 joerg Exp $");
 #endif /* not lint */
 
 #include <sys/types.h>
@@ -161,7 +161,7 @@
 				*where32 = tmp32;
 			rdbg(("32/32S %s in %s --> %p in %s",
 			    obj->strtab + obj->symtab[symnum].st_name,
-			    obj->path, (void *)(unsigned long)*where32,
+			    obj->path, (void *)(uintptr_t)*where32,
 			    defobj->path));
 			break;
 		case R_TYPE(64):	/* word64 S + A */
@@ -211,6 +211,50 @@
 			    (void *)*where64));
        			break;
 
+		case R_TYPE(TPOFF64):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			if (!defobj->tls_done &&
+			    _rtld_tls_offset_allocate(obj))
+				return -1;
+
+			*where64 = (Elf64_Addr)(def->st_value -
+			    defobj->tlsoffset + rela->r_addend);
+
+			rdbg(("TPOFF64 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where64));
+
+			break;
+
+		case R_TYPE(DTPMOD64):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			*where64 = (Elf64_Addr)defobj->tlsindex;
+
+			rdbg(("DTPMOD64 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where64));
+
+			break;
+
+		case R_TYPE(DTPOFF64):
+			def = _rtld_find_symdef(symnum, obj, &defobj, false);
+			if (def == NULL)
+				return -1;
+
+			*where64 = (Elf64_Addr)(def->st_value + rela->r_addend);
+
+			rdbg(("DTPOFF64 %s in %s --> %p",
+			    obj->strtab + obj->symtab[symnum].st_name,
+			    obj->path, (void *)*where64));
+
+			break;
+
 		case R_TYPE(COPY):
 			rdbg(("COPY"));
 			break;

Index: src/sys/arch/amd64/include/types.h
diff -u src/sys/arch/amd64/include/types.h:1.36 src/sys/arch/amd64/include/types.h:1.37
--- src/sys/arch/amd64/include/types.h:1.36	Thu Feb 24 04:28:45 2011
+++ src/sys/arch/amd64/include/types.h	Sat Mar 12 22:54:37 2011
@@ -1,4 +1,4 @@
-/*	$NetBSD: types.h,v 1.36 2011/02/24 04:28:45 joerg Exp $	*/
+/*	$NetBSD: types.h,v 1.37 2011/03/12 22:54:37 joerg Exp $	*/
 
 /*-
  * Copyright (c) 1990 The Regents of the University of California.
@@ -86,6 +86,8 @@
 #define	__HAVE_ATOMIC_AS_MEMBAR
 #define	__HAVE_CPU_LWP_SETPRIVATE
 #define	__HAVE___LWP_GETPRIVATE_FAST
+#define	__HAVE_TLS_VARIANT_II
+#define	__HAVE_COMMON___TLS_GET_ADDR
 #define	__HAVE_INTR_CONTROL
 
 #ifdef _KERNEL_OPT

Index: src/sys/arch/i386/include/types.h
diff -u src/sys/arch/i386/include/types.h:1.71 src/sys/arch/i386/include/types.h:1.72
--- src/sys/arch/i386/include/types.h:1.71	Thu Feb 24 04:28:46 2011
+++ src/sys/arch/i386/include/types.h	Sat Mar 12 22:54:37 2011
@@ -1,4 +1,4 @@
-/*	$NetBSD: types.h,v 1.71 2011/02/24 04:28:46 joerg Exp $	*/
+/*	$NetBSD: types.h,v 1.72 2011/03/12 22:54:37 joerg Exp $	*/
 
 /*-
  * Copyright (c) 1990 The Regents of the University of California.
@@ -113,6 +113,8 @@
 #define	__HAVE_CPU_LWP_SETPRIVATE
 #define	__HAVE_INTR_CONTROL
 #define	__HAVE___LWP_GETPRIVATE_FAST
+#define	__HAVE_TLS_VARIANT_II
+#define	__HAVE_COMMON___TLS_GET_ADDR
 
 #if defined(_KERNEL)
 #define	__HAVE_RAS

Index: src/sys/arch/sh3/include/types.h
diff -u src/sys/arch/sh3/include/types.h:1.31 src/sys/arch/sh3/include/types.h:1.32
--- src/sys/arch/sh3/include/types.h:1.31	Thu Feb 24 04:28:47 2011
+++ src/sys/arch/sh3/include/types.h	Sat Mar 12 22:54:37 2011
@@ -1,4 +1,4 @@
-/*	$NetBSD: types.h,v 1.31 2011/02/24 04:28:47 joerg Exp $	*/
+/*	$NetBSD: types.h,v 1.32 2011/03/12 22:54:37 joerg Exp $	*/
 
 /*-
  * Copyright (c) 1990 The Regents of the University of California.
@@ -79,5 +79,7 @@
 
 #define	__HAVE_CPU_LWP_SETPRIVATE
 #define	__HAVE___LWP_GETPRIVATE_FAST
+#define	__HAVE_COMMON___TLS_GET_ADDR
+#define	__HAVE_TLS_VARIANT_I
 
 #endif	/* !_SH3_TYPES_H_ */

Reply via email to