svn commit: r195745 - in head: include lib/libc/gen libexec/rtld-elf

Konstantin Belousov kib at FreeBSD.org
Fri Jul 17 19:45:43 UTC 2009


Author: kib
Date: Fri Jul 17 19:45:42 2009
New Revision: 195745
URL: http://svn.freebsd.org/changeset/base/195745

Log:
  Implement RTLD_NOLOAD flag for dlopen(3).
  
  Requested and tested by:	jkim
  Reviewed by:	kan
  Approved by:	re (kensmith)

Modified:
  head/include/dlfcn.h
  head/lib/libc/gen/dlopen.3
  head/libexec/rtld-elf/rtld.c

Modified: head/include/dlfcn.h
==============================================================================
--- head/include/dlfcn.h	Fri Jul 17 19:38:07 2009	(r195744)
+++ head/include/dlfcn.h	Fri Jul 17 19:45:42 2009	(r195745)
@@ -48,6 +48,7 @@
 #define	RTLD_LOCAL	0	/* Opposite of RTLD_GLOBAL, and the default. */
 #define	RTLD_TRACE	0x200	/* Trace loaded objects and exit. */
 #define	RTLD_NODELETE	0x01000	/* Do not remove members. */
+#define	RTLD_NOLOAD	0x02000	/* Do not load if not already loaded. */
 
 /*
  * Request arguments for dlinfo().

Modified: head/lib/libc/gen/dlopen.3
==============================================================================
--- head/lib/libc/gen/dlopen.3	Fri Jul 17 19:38:07 2009	(r195744)
+++ head/lib/libc/gen/dlopen.3	Fri Jul 17 19:45:42 2009	(r195745)
@@ -32,7 +32,7 @@
 .\" @(#) dlopen.3 1.6 90/01/31 SMI
 .\" $FreeBSD$
 .\"
-.Dd April 1, 2009
+.Dd July 7, 2009
 .Os
 .Dt DLOPEN 3
 .Sh NAME
@@ -148,6 +148,13 @@ The same behaviour may be requested by
 .Fl "z nodelete"
 option of the static linker
 .Xr ld 1 .
+.It Dv RTLD_NOLOAD
+Ony return valid handle for the object if it is already loaded in
+the process address space, otherwise
+.Dv NULL
+is returned.
+Other mode flags may be specified, which will be applied for promotion
+for the found object.
 .El
 .Pp
 If

Modified: head/libexec/rtld-elf/rtld.c
==============================================================================
--- head/libexec/rtld-elf/rtld.c	Fri Jul 17 19:38:07 2009	(r195744)
+++ head/libexec/rtld-elf/rtld.c	Fri Jul 17 19:45:42 2009	(r195745)
@@ -105,7 +105,7 @@ static void linkmap_add(Obj_Entry *);
 static void linkmap_delete(Obj_Entry *);
 static int load_needed_objects(Obj_Entry *);
 static int load_preload_objects(void);
-static Obj_Entry *load_object(const char *, const Obj_Entry *);
+static Obj_Entry *load_object(const char *, const Obj_Entry *, int);
 static Obj_Entry *obj_from_addr(const void *);
 static void objlist_call_fini(Objlist *, bool, int *);
 static void objlist_call_init(Objlist *, int *);
@@ -1432,7 +1432,8 @@ load_needed_objects(Obj_Entry *first)
 	Needed_Entry *needed;
 
 	for (needed = obj->needed;  needed != NULL;  needed = needed->next) {
-	    obj1 = needed->obj = load_object(obj->strtab + needed->name, obj);
+	    obj1 = needed->obj = load_object(obj->strtab + needed->name, obj,
+		false);
 	    if (obj1 == NULL && !ld_tracing)
 		return -1;
 	    if (obj1 != NULL && obj1->z_nodelete && !obj1->ref_nodel) {
@@ -1463,7 +1464,7 @@ load_preload_objects(void)
 
 	savech = p[len];
 	p[len] = '\0';
-	if (load_object(p, NULL) == NULL)
+	if (load_object(p, NULL, false) == NULL)
 	    return -1;	/* XXX - cleanup */
 	p[len] = savech;
 	p += len;
@@ -1480,7 +1481,7 @@ load_preload_objects(void)
  * on failure.
  */
 static Obj_Entry *
-load_object(const char *name, const Obj_Entry *refobj)
+load_object(const char *name, const Obj_Entry *refobj, int noload)
 {
     Obj_Entry *obj;
     int fd = -1;
@@ -1526,6 +1527,8 @@ load_object(const char *name, const Obj_
 	close(fd);
 	return obj;
     }
+    if (noload)
+	return (NULL);
 
     /* First use of this object, so we must map it in */
     obj = do_load_object(fd, name, path, &sb);
@@ -1982,13 +1985,14 @@ dlopen(const char *name, int mode)
     Obj_Entry **old_obj_tail;
     Obj_Entry *obj;
     Objlist initlist;
-    int result, lockstate, nodelete;
+    int result, lockstate, nodelete, noload;
 
     LD_UTRACE(UTRACE_DLOPEN_START, NULL, NULL, 0, mode, name);
     ld_tracing = (mode & RTLD_TRACE) == 0 ? NULL : "1";
     if (ld_tracing != NULL)
 	environ = (char **)*get_program_var_addr("environ");
     nodelete = mode & RTLD_NODELETE;
+    noload = mode & RTLD_NOLOAD;
 
     objlist_init(&initlist);
 
@@ -2001,7 +2005,7 @@ dlopen(const char *name, int mode)
 	obj = obj_main;
 	obj->refcount++;
     } else {
-	obj = load_object(name, obj_main);
+	obj = load_object(name, obj_main, noload);
     }
 
     if (obj) {


More information about the svn-src-all mailing list