mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 12:32:34 +00:00 
			
		
		
		
	 e1ff30a360
			
		
	
	
		e1ff30a360
		
	
	
	
	
		
			
			This is not technically a toolchain change, but it does require rebuilding the toolchain for x86_64 (and just that).
		
			
				
	
	
		
			74 lines
		
	
	
	
		
			1.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			74 lines
		
	
	
	
		
			1.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Gunnar Beutner <gunnar@beutner.name>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/String.h>
 | |
| #include <AK/Types.h>
 | |
| #include <LibDl/dlfcn.h>
 | |
| #include <LibDl/dlfcn_integration.h>
 | |
| #include <string.h>
 | |
| 
 | |
| // FIXME: use thread_local and a String once TLS works
 | |
| __thread char* s_dlerror_text = NULL;
 | |
| __thread bool s_dlerror_retrieved = false;
 | |
| 
 | |
| static void store_error(const String& error)
 | |
| {
 | |
|     free(s_dlerror_text);
 | |
|     s_dlerror_text = strdup(error.characters());
 | |
|     s_dlerror_retrieved = false;
 | |
| }
 | |
| 
 | |
| int dlclose(void* handle)
 | |
| {
 | |
|     auto result = __dlclose(handle);
 | |
|     if (result.is_error()) {
 | |
|         store_error(result.error().text);
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| char* dlerror()
 | |
| {
 | |
|     if (s_dlerror_retrieved) {
 | |
|         free(s_dlerror_text);
 | |
|         s_dlerror_text = nullptr;
 | |
|     }
 | |
|     s_dlerror_retrieved = true;
 | |
|     return const_cast<char*>(s_dlerror_text);
 | |
| }
 | |
| 
 | |
| void* dlopen(const char* filename, int flags)
 | |
| {
 | |
|     auto result = __dlopen(filename, flags);
 | |
|     if (result.is_error()) {
 | |
|         store_error(result.error().text);
 | |
|         return nullptr;
 | |
|     }
 | |
|     return result.value();
 | |
| }
 | |
| 
 | |
| void* dlsym(void* handle, const char* symbol_name)
 | |
| {
 | |
|     auto result = __dlsym(handle, symbol_name);
 | |
|     if (result.is_error()) {
 | |
|         store_error(result.error().text);
 | |
|         return nullptr;
 | |
|     }
 | |
|     return result.value();
 | |
| }
 | |
| 
 | |
| int dladdr(void* addr, Dl_info* info)
 | |
| {
 | |
|     auto result = __dladdr(addr, info);
 | |
|     if (result.is_error()) {
 | |
|         // FIXME: According to the man page glibc does _not_ make the error
 | |
|         // available via dlerror(), however we do. Does this break anything?
 | |
|         store_error(result.error().text);
 | |
|         return 0;
 | |
|     }
 | |
|     return 1;
 | |
| }
 |