mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-31 11:03:14 +00:00 
			
		
		
		
	 dd336c554d
			
		
	
	
		dd336c554d
		
	
	
	
	
		
			
			fix memory leak introduced by the patch 6e03a201bb:
firmware: speed up request_firmware()
1. vfree won't release pages there were allocated explicitly and mapped
using vmap. The memory has to be vunmap-ed and the pages needs
to be freed explicitly
2. page array is moved into the 'struct
firmware' so that we can free it from release_firmware()
and not only in fw_dev_release()
The fix doesn't break the firmware load speed.
Cc: Johannes Berg <johannes@sipsolutions.net>
Cc: Ming Lei <tom.leiming@gmail.com>
Cc: Catalin Marinas <catalin.marinas@arm.com>
Singed-off-by: Kay Sievers <kay.sievers@vrfy.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Signed-off-by: Tomas Winkler <tomas.winkler@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
		
	
			
		
			
				
	
	
		
			68 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			68 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef _LINUX_FIRMWARE_H
 | |
| #define _LINUX_FIRMWARE_H
 | |
| 
 | |
| #include <linux/module.h>
 | |
| #include <linux/types.h>
 | |
| #include <linux/compiler.h>
 | |
| #include <linux/gfp.h>
 | |
| 
 | |
| #define FW_ACTION_NOHOTPLUG 0
 | |
| #define FW_ACTION_HOTPLUG 1
 | |
| 
 | |
| struct firmware {
 | |
| 	size_t size;
 | |
| 	const u8 *data;
 | |
| 	struct page **pages;
 | |
| };
 | |
| 
 | |
| struct device;
 | |
| 
 | |
| struct builtin_fw {
 | |
| 	char *name;
 | |
| 	void *data;
 | |
| 	unsigned long size;
 | |
| };
 | |
| 
 | |
| /* We have to play tricks here much like stringify() to get the
 | |
|    __COUNTER__ macro to be expanded as we want it */
 | |
| #define __fw_concat1(x, y) x##y
 | |
| #define __fw_concat(x, y) __fw_concat1(x, y)
 | |
| 
 | |
| #define DECLARE_BUILTIN_FIRMWARE(name, blob)				     \
 | |
| 	DECLARE_BUILTIN_FIRMWARE_SIZE(name, &(blob), sizeof(blob))
 | |
| 
 | |
| #define DECLARE_BUILTIN_FIRMWARE_SIZE(name, blob, size)			     \
 | |
| 	static const struct builtin_fw __fw_concat(__builtin_fw,__COUNTER__) \
 | |
| 	__used __section(.builtin_fw) = { name, blob, size }
 | |
| 
 | |
| #if defined(CONFIG_FW_LOADER) || (defined(CONFIG_FW_LOADER_MODULE) && defined(MODULE))
 | |
| int request_firmware(const struct firmware **fw, const char *name,
 | |
| 		     struct device *device);
 | |
| int request_firmware_nowait(
 | |
| 	struct module *module, int uevent,
 | |
| 	const char *name, struct device *device, gfp_t gfp, void *context,
 | |
| 	void (*cont)(const struct firmware *fw, void *context));
 | |
| 
 | |
| void release_firmware(const struct firmware *fw);
 | |
| #else
 | |
| static inline int request_firmware(const struct firmware **fw,
 | |
| 				   const char *name,
 | |
| 				   struct device *device)
 | |
| {
 | |
| 	return -EINVAL;
 | |
| }
 | |
| static inline int request_firmware_nowait(
 | |
| 	struct module *module, int uevent,
 | |
| 	const char *name, struct device *device, gfp_t gfp, void *context,
 | |
| 	void (*cont)(const struct firmware *fw, void *context))
 | |
| {
 | |
| 	return -EINVAL;
 | |
| }
 | |
| 
 | |
| static inline void release_firmware(const struct firmware *fw)
 | |
| {
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #endif
 |