Skip site navigation (1)Skip section navigation (2)
Date:      Sat, 4 Nov 2006 03:27:00 GMT
From:      John Birrell <jb@FreeBSD.org>
To:        Perforce Change Reviews <perforce@freebsd.org>
Subject:   PERFORCE change 109209 for review
Message-ID:  <200611040327.kA43R0r9073857@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=109209

Change 109209 by jb@jb_freebsd8 on 2006/11/04 03:26:42

	IFlibbsdelf

Affected files ...

.. //depot/projects/dtrace/src/lib/libelf/Makefile#17 integrate
.. //depot/projects/dtrace/src/lib/libelf/_libelf.h#11 integrate
.. //depot/projects/dtrace/src/lib/libelf/elf_getscn.3#3 integrate
.. //depot/projects/dtrace/src/lib/libelf/elf_scn.c#6 integrate
.. //depot/projects/dtrace/src/lib/libelf/elf_shnum.c#2 integrate
.. //depot/projects/dtrace/src/lib/libelf/elf_update.c#6 integrate
.. //depot/projects/dtrace/src/lib/libelf/gelf_getshdr.c#5 integrate
.. //depot/projects/dtrace/src/lib/libelf/libelf_shdr.c#1 branch

Differences ...

==== //depot/projects/dtrace/src/lib/libelf/Makefile#17 (text+ko) ====

@@ -43,6 +43,7 @@
 	libelf_data.c						\
 	libelf_ehdr.c						\
 	libelf_phdr.c						\
+	libelf_shdr.c						\
 	libelf_xlate.c
 INCS=	libelf.h gelf.h
 

==== //depot/projects/dtrace/src/lib/libelf/_libelf.h#11 (text+ko) ====

@@ -148,7 +148,8 @@
 void	(*_libelf_get_translator(Elf_Type _t, int _direction, int _elfclass))
 	    (char *_dst, char *_src, size_t _cnt, int _byteswap);
 void	*_libelf_getphdr(Elf *_e, int _elfclass);
-int	_libelf_getshnum(void *_eh, int _elfclass, size_t *_shnum);
+void	*_libelf_getshdr(Elf_Scn *_scn, int _elfclass);
+int	_libelf_getshnum(Elf *_e, void *_eh, int _elfclass, size_t *_shnum);
 int	_libelf_getshstrndx(void *_eh, int _elfclass, size_t *_shstrndx);
 void	_libelf_init_elf(Elf *_e, Elf_Kind _kind);
 int	_libelf_malign(Elf_Type _t, int _elfclass);
@@ -157,7 +158,7 @@
 Elf_Data *_libelf_release_data(Elf_Data *_d);
 Elf	*_libelf_release_elf(Elf *_e);
 Elf_Scn	*_libelf_release_scn(Elf_Scn *_s);
-int	_libelf_setshnum(void *_eh, int _elfclass, size_t _shnum);
+int	_libelf_setshnum(Elf *_e, void *_eh, int _elfclass, size_t _shnum);
 int	_libelf_setshstrndx(void *_eh, int _elfclass, size_t _shstrndx);
 Elf_Data *_libelf_xlate(Elf_Data *_d, const Elf_Data *_s,
     unsigned int _encoding, int _elfclass, int _direction);

==== //depot/projects/dtrace/src/lib/libelf/elf_getscn.3#3 (text+ko) ====

@@ -133,6 +133,14 @@
 Descriptor
 .Ar elf
 was of an unknown ELF class.
+.It Bq Er ELF_E_SECTION
+Argument
+.Ar elf
+specified extended section numbering in the ELF header with the section header at
+index
+.Dv SHN_UNDEF
+not being of type
+.Dv SHT_NULL .
 .El
 .Sh SEE ALSO
 .Xr elf 3 ,

==== //depot/projects/dtrace/src/lib/libelf/elf_scn.c#6 (text+ko) ====

@@ -41,10 +41,11 @@
 static int
 _libelf_load_scn(Elf *e, void *ehdr)
 {
-	int ec;
+	int ec, swapbytes;
 	size_t fsz, i, shnum;
 	uint64_t shoff;
-	char *src, *dst;
+	uint32_t shtype;
+	char *src;
 	Elf32_Ehdr *eh32;
 	Elf64_Ehdr *eh64;
 	Elf_Scn *scn;
@@ -79,20 +80,43 @@
 
 	xlator = _libelf_get_translator(ELF_T_SHDR, ELF_TOMEMORY, ec);
 
+	swapbytes = e->e_byteorder != LIBELF_PRIVATE(byteorder);
 	src = e->e_rawfile + shoff;
+	i = 0;
+
+	if (shnum == (size_t) 0 && shoff != 0LL) {
+		/* Extended section numbering */
+		if ((scn = _libelf_allocate_scn(e, (size_t) 0)) == NULL)
+			return (0);
+
+		(*xlator)((char *) &scn->s_shdr, src, (size_t) 1, swapbytes);
+
+		if (ec == ELFCLASS32) {
+			shtype = scn->s_shdr.s_shdr32.sh_type;
+			shnum = scn->s_shdr.s_shdr32.sh_size;
+		} else {
+			shtype = scn->s_shdr.s_shdr64.sh_type;
+			shnum = scn->s_shdr.s_shdr64.sh_size;
+		}
+
+		if (shtype != SHT_NULL) {
+			LIBELF_SET_ERROR(SECTION, 0);
+			return (0);
+		}
+
+		scn->s_size = 0LL;
+		scn->s_offset = scn->s_rawoff = 0LL;
 
-	for (i = 0; i < shnum; i++, src += fsz) {
+		i++;
+		src += fsz;
+	}
+
+	for (; i < shnum; i++, src += fsz) {
 		if ((scn = _libelf_allocate_scn(e, i)) == NULL)
 			return (0);
 
-		if (ec == ELFCLASS32)
-			dst = (char *) &scn->s_shdr.s_shdr32;
-		else
-			dst = (char *) &scn->s_shdr.s_shdr64;
+		(*xlator)((char *) &scn->s_shdr, src, (size_t) 1, swapbytes);
 
-		(*xlator)(dst, src, (size_t) 1,
-		    e->e_byteorder != LIBELF_PRIVATE(byteorder));
-
 		if (ec == ELFCLASS32) {
 			scn->s_offset = scn->s_rawoff =
 			    scn->s_shdr.s_shdr32.sh_offset;
@@ -112,7 +136,6 @@
 {
 	int ec;
 	void *ehdr;
-	uint16_t *pshnum;
 	Elf_Scn *s;
 
 	if (e == NULL || e->e_kind != ELF_K_ELF ||
@@ -124,16 +147,6 @@
 	if ((ehdr = _libelf_ehdr(e, ec, 0)) == NULL)
 		return (NULL);
 
-	if (ec == ELFCLASS32)
-		pshnum = &((Elf32_Ehdr *) ehdr)->e_shnum;
-	else
-		pshnum = &((Elf64_Ehdr *) ehdr)->e_shnum;
-
-	if (index >= *pshnum) {
-		LIBELF_SET_ERROR(ARGUMENT, 0);
-		return (NULL);
-	}
-
 	if (e->e_cmd != ELF_C_WRITE && STAILQ_EMPTY(&e->e_u.e_elf.e_scn) &&
 	    _libelf_load_scn(e, ehdr) == 0)
 		return (NULL);
@@ -142,6 +155,7 @@
 		if (s->s_ndx == index)
 			return (s);
 
+	LIBELF_SET_ERROR(ARGUMENT, 0);
 	return (NULL);
 }
 
@@ -190,7 +204,7 @@
 	    _libelf_load_scn(e, ehdr) == 0)
 		return (NULL);
 
-	if (_libelf_getshnum(ehdr, ec, &shnum) == 0)
+	if (_libelf_getshnum(e, ehdr, ec, &shnum) == 0)
 		return (NULL);
 
 	if (STAILQ_EMPTY(&e->e_u.e_elf.e_scn)) {
@@ -208,7 +222,7 @@
 
 	shnum++;
 
-	if (_libelf_setshnum(ehdr, ec, shnum) == 0)
+	if (_libelf_setshnum(e, ehdr, ec, shnum) == 0)
 		return (NULL);
 
 	(void) elf_flagscn(scn, ELF_C_SET, ELF_F_DIRTY);

==== //depot/projects/dtrace/src/lib/libelf/elf_shnum.c#2 (text+ko) ====

@@ -36,54 +36,75 @@
 #include "_libelf.h"
 
 int
-_libelf_getshnum(void *eh, int ec, size_t *shnum)
+_libelf_getshnum(Elf *e, void *eh, int ec, size_t *shnum)
 {
-	Elf32_Ehdr *eh32;
-	Elf64_Ehdr *eh64;
 	Elf64_Off off;
+	Elf_Scn *scn;
+	void *sh;
 	size_t n;
 
-	eh32 = eh;
-	eh64 = eh;
 	if (ec == ELFCLASS32) {
-		n = eh32->e_shnum;
-		off = (Elf64_Off) eh32->e_shoff;
+		n = ((Elf32_Ehdr *) eh)->e_shnum;
+		off = (Elf64_Off) ((Elf32_Ehdr *) eh)->e_shoff;
 	} else {
-		n = eh64->e_shnum;
-		off = eh64->e_shoff;
+		n = ((Elf64_Ehdr *) eh)->e_shnum;
+		off = ((Elf64_Ehdr *) eh)->e_shoff;
+	}
+
+	if (n != 0) {
+		*shnum = n;
+		return (1);
+	}
+
+	if (off == 0L) {
+		*shnum = (size_t) 0;
+		return (1);
 	}
 
-	if (n == 0)
-		if (off != 0) {
-			/* TODO */
-			LIBELF_SET_ERROR(UNIMPL, 0);
-			return (0);
-		} else
-			*shnum = 0;
+	/*
+	 * If 'e_shnum' is zero and 'e_shoff' is non-zero, the file is
+	 * using extended section numbering, and the true section
+	 * number is kept in the 'sh_size' field of the section header
+	 * at offset SHN_UNDEF.
+	 */
+	if ((scn = elf_getscn(e, SHN_UNDEF)) == NULL)
+		return (0);
+	if ((sh = _libelf_getshdr(scn, ec)) == NULL)
+		return (0);
+
+	if (ec == ELFCLASS32)
+		*shnum = ((Elf32_Shdr *) sh)->sh_size;
 	else
-		*shnum = n;
+		*shnum = ((Elf64_Shdr *) sh)->sh_size;
 
 	return (1);
 }
 
 int
-_libelf_setshnum(void *eh, int ec, size_t shnum)
+_libelf_setshnum(Elf *e, void *eh, int ec, size_t shnum)
 {
-	Elf32_Ehdr *eh32;
-	Elf64_Ehdr *eh64;
+	Elf_Scn *scn;
+	void *sh;
 
-	eh32 = eh;
-	eh64 = eh;
+	if (shnum < SHN_LORESERVE) {
+		if (ec == ELFCLASS32)
+			((Elf32_Ehdr *) eh)->e_shnum = shnum;
+		else
+			((Elf64_Ehdr *) eh)->e_shnum = shnum;
+		return (1);
+	}
 
-	if (shnum >= SHN_XINDEX) {
-		LIBELF_SET_ERROR(UNIMPL, 0);
+	if ((scn = elf_getscn(e, SHN_UNDEF)) == NULL)
+		return (0);
+	if ((sh = _libelf_getshdr(scn, ec)) == NULL)
 		return (0);
-	}
-	if (ec == ELFCLASS32) {
-		eh32->e_shnum = shnum;
-	} else {
-		eh64->e_shnum = shnum;
-	}
+
+	if (ec == ELFCLASS32)
+		((Elf32_Shdr *) sh)->sh_size = shnum;
+	else
+		((Elf64_Shdr *) sh)->sh_size = shnum;
+
+	(void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY);
 
 	return (1);
 }
@@ -101,5 +122,5 @@
 		return (0);
 	}
 
-	return (_libelf_getshnum(eh, ec, shnum));
+	return (_libelf_getshnum(e, eh, ec, shnum));
 }

==== //depot/projects/dtrace/src/lib/libelf/elf_update.c#6 (text+ko) ====

@@ -76,6 +76,7 @@
 	int ec;
 	Elf_Data *d, *td;
 	unsigned int elftype;
+	uint32_t sh_type;
 	uint64_t sh_align, sh_entsize, sh_offset, sh_size;
 	uint64_t scn_size, scn_alignment;
 
@@ -101,19 +102,23 @@
 	ec = e->e_class;
 
 	if (ec == ELFCLASS32) {
-		elftype    = _libelf_xlate_shtype(s->s_shdr.s_shdr32.sh_type);
+		sh_type    = s->s_shdr.s_shdr32.sh_type;
 		sh_align   = (uint64_t) s->s_shdr.s_shdr32.sh_addralign;
 		sh_entsize = (uint64_t) s->s_shdr.s_shdr32.sh_entsize;
 		sh_offset  = (uint64_t) s->s_shdr.s_shdr32.sh_offset;
 		sh_size    = (uint64_t) s->s_shdr.s_shdr32.sh_size;
 	} else {
-		elftype    = _libelf_xlate_shtype(s->s_shdr.s_shdr64.sh_type);
+		sh_type    = s->s_shdr.s_shdr64.sh_type;
 		sh_align   = s->s_shdr.s_shdr64.sh_addralign;
 		sh_entsize = s->s_shdr.s_shdr64.sh_entsize;
 		sh_offset  = s->s_shdr.s_shdr64.sh_offset;
 		sh_size    = s->s_shdr.s_shdr64.sh_size;
 	}
 
+	if (sh_type == SHT_NULL || sh_type == SHT_NOBITS)
+		return (1);
+
+	elftype = _libelf_xlate_shtype(sh_type);
 	if (elftype > ELF_T_LAST) {
 		LIBELF_SET_ERROR(SECTION, 0);
 		return (0);
@@ -356,7 +361,7 @@
 		return ((off_t) -1);
 	}
 
-	if (_libelf_getshnum(ehdr, ec, &shnum) == 0)
+	if (_libelf_getshnum(e, ehdr, ec, &shnum) == 0)
 		return ((off_t) -1);
 
 	e->e_byteorder = eh_byteorder;

==== //depot/projects/dtrace/src/lib/libelf/gelf_getshdr.c#5 (text+ko) ====

@@ -36,69 +36,36 @@
 Elf32_Shdr *
 elf32_getshdr(Elf_Scn *s)
 {
-	int ec;
-	Elf *e;
-
-	if (s == NULL || (e = s->s_elf) == NULL ||
-	    e->e_kind != ELF_K_ELF) {
-		LIBELF_SET_ERROR(ARGUMENT, 0);
-		return (NULL);
-	}
-
-	ec = e->e_class;
-
-	assert(ec == ELFCLASS32 || ec == ELFCLASS64);
-
-	if (ec != ELFCLASS32) {
-		LIBELF_SET_ERROR(CLASS, 0);
-		return (NULL);
-	}
-
-	return (&s->s_shdr.s_shdr32);
+	return (_libelf_getshdr(s, ELFCLASS32));
 }
 
 Elf64_Shdr *
 elf64_getshdr(Elf_Scn *s)
 {
-	int ec;
-	Elf *e;
-
-	if (s == NULL || (e = s->s_elf) == NULL ||
-	    e->e_kind != ELF_K_ELF) {
-		LIBELF_SET_ERROR(ARGUMENT, 0);
-		return (NULL);
-	}
-
-	ec = e->e_class;
-
-	assert(ec == ELFCLASS32 || ec == ELFCLASS64);
-
-	if (ec != ELFCLASS64) {
-		LIBELF_SET_ERROR(CLASS, 0);
-		return (NULL);
-	}
-
-	return (&s->s_shdr.s_shdr64);
+	return (_libelf_getshdr(s, ELFCLASS64));
 }
 
 GElf_Shdr *
 gelf_getshdr(Elf_Scn *s, GElf_Shdr *d)
 {
 	int ec;
-	Elf *e;
+	void *sh;
 	Elf32_Shdr *sh32;
+	Elf64_Shdr *sh64;
 
-	if (s == NULL || d == NULL ||
-	    (e = s->s_elf) == NULL || e->e_kind != ELF_K_ELF) {
+	if (d == NULL) {
 		LIBELF_SET_ERROR(ARGUMENT, 0);
 		return (NULL);
 	}
 
-	ec = e->e_class;
+	if ((sh = _libelf_getshdr(s, ELFCLASSNONE)) == NULL)
+		return (NULL);
+
+	ec = s->s_elf->e_class;
 	assert(ec == ELFCLASS32 || ec == ELFCLASS64);
 
 	if (ec == ELFCLASS32) {
-		sh32 = &s->s_shdr.s_shdr32;
+		sh32 = (Elf32_Shdr *) sh;
 
 		d->sh_name      = sh32->sh_name;
 		d->sh_type      = sh32->sh_type;
@@ -110,8 +77,10 @@
 		d->sh_info      = sh32->sh_info;
 		d->sh_addralign = (Elf64_Xword) sh32->sh_addralign;
 		d->sh_entsize   = (Elf64_Xword) sh32->sh_entsize;
-	} else
-		*d = s->s_shdr.s_shdr64;
+	} else {
+		sh64 = (Elf64_Shdr *) sh;
+		*d = *sh64;
+	}
 
 	return (d);
 }



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200611040327.kA43R0r9073857>