Merge git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6.git
Conflicts: sound/pci/Kconfig
This commit is contained in:
@@ -1,18 +1,39 @@
|
||||
#ifndef _LINUX_FIRMWARE_H
|
||||
#define _LINUX_FIRMWARE_H
|
||||
|
||||
#include <linux/module.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/compiler.h>
|
||||
|
||||
#define FIRMWARE_NAME_MAX 30
|
||||
#define FW_ACTION_NOHOTPLUG 0
|
||||
#define FW_ACTION_HOTPLUG 1
|
||||
|
||||
struct firmware {
|
||||
size_t size;
|
||||
u8 *data;
|
||||
const u8 *data;
|
||||
};
|
||||
|
||||
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);
|
||||
|
74
include/linux/ihex.h
Normal file
74
include/linux/ihex.h
Normal file
@@ -0,0 +1,74 @@
|
||||
/*
|
||||
* Compact binary representation of ihex records. Some devices need their
|
||||
* firmware loaded in strange orders rather than a single big blob, but
|
||||
* actually parsing ihex-as-text within the kernel seems silly. Thus,...
|
||||
*/
|
||||
|
||||
#ifndef __LINUX_IHEX_H__
|
||||
#define __LINUX_IHEX_H__
|
||||
|
||||
#include <linux/types.h>
|
||||
#include <linux/firmware.h>
|
||||
#include <linux/device.h>
|
||||
|
||||
/* Intel HEX files actually limit the length to 256 bytes, but we have
|
||||
drivers which would benefit from using separate records which are
|
||||
longer than that, so we extend to 16 bits of length */
|
||||
struct ihex_binrec {
|
||||
__be32 addr;
|
||||
__be16 len;
|
||||
uint8_t data[0];
|
||||
} __attribute__((aligned(4)));
|
||||
|
||||
/* Find the next record, taking into account the 4-byte alignment */
|
||||
static inline const struct ihex_binrec *
|
||||
ihex_next_binrec(const struct ihex_binrec *rec)
|
||||
{
|
||||
int next = ((be16_to_cpu(rec->len) + 5) & ~3) - 2;
|
||||
rec = (void *)&rec->data[next];
|
||||
|
||||
return be16_to_cpu(rec->len) ? rec : NULL;
|
||||
}
|
||||
|
||||
/* Check that ihex_next_binrec() won't take us off the end of the image... */
|
||||
static inline int ihex_validate_fw(const struct firmware *fw)
|
||||
{
|
||||
const struct ihex_binrec *rec;
|
||||
size_t ofs = 0;
|
||||
|
||||
while (ofs <= fw->size - sizeof(*rec)) {
|
||||
rec = (void *)&fw->data[ofs];
|
||||
|
||||
/* Zero length marks end of records */
|
||||
if (!be16_to_cpu(rec->len))
|
||||
return 0;
|
||||
|
||||
/* Point to next record... */
|
||||
ofs += (sizeof(*rec) + be16_to_cpu(rec->len) + 3) & ~3;
|
||||
}
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* Request firmware and validate it so that we can trust we won't
|
||||
* run off the end while reading records... */
|
||||
static inline int request_ihex_firmware(const struct firmware **fw,
|
||||
const char *fw_name,
|
||||
struct device *dev)
|
||||
{
|
||||
const struct firmware *lfw;
|
||||
int ret;
|
||||
|
||||
ret = request_firmware(&lfw, fw_name, dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
ret = ihex_validate_fw(lfw);
|
||||
if (ret) {
|
||||
dev_err(dev, "Firmware \"%s\" not valid IHEX records\n",
|
||||
fw_name);
|
||||
release_firmware(lfw);
|
||||
return ret;
|
||||
}
|
||||
*fw = lfw;
|
||||
return 0;
|
||||
}
|
||||
#endif /* __LINUX_IHEX_H__ */
|
Reference in New Issue
Block a user