depends on OMAP_DEBUG_DEVICES
default y if LEDS || LEDS_OMAP_DEBUG
+config OMAP_DEBUG_SRAM_PATCH
+ bool "Extra sanity checking for SRAM patch code"
+ depends on ARCH_OMAP
+ default y
+ help
+ Say Y here if you want the kernel to use extra caution
+ in patching SRAM virtual addresses. If you are
+ confident in your SRAM code, disabling this will save
+ about 600 bytes.
+
config OMAP_RESET_CLOCKS
bool "Reset unused clocks during boot"
depends on ARCH_OMAP
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
+#undef DEBUG
#include <linux/module.h>
#include <linux/kernel.h>
return (void *)omap_sram_ceil;
}
+/**
+ * omap_sram_patch_va - patch a virtual address into SRAM code
+ * @srcfn: original start address (in DRAM) of function to patch
+ * @srcd: original address (in DRAM) of location to patch
+ * @sramfn: start address (in SRAM) of function to patch
+ * @d: virtual address to insert
+ *
+ * Replace a location in SRAM containing a magic number
+ * (SRAM_VA_MAGIC) with a caller-specified virtual address. Used to
+ * dynamically patch SRAM code at runtime for multiboot, since some
+ * register addresses change depending on the OMAP chip in use.
+ * Returns 1 upon success, 0 upon failure.
+ */
+int omap_sram_patch_va(void *srcfn, void *srcd, void *sramfn, void __iomem *d)
+{
+ unsigned long sram_addr;
+ long offs;
+
+ offs = (unsigned long)srcd - (unsigned long)srcfn;
+ sram_addr = (unsigned long)sramfn + offs;
+
+#ifdef CONFIG_OMAP_DEBUG_SRAM_PATCH
+ if (offs < 0) {
+ printk(KERN_ERR "sram: patch address 0x%0lx < function start "
+ "address 0x%0lx\n", (unsigned long)srcd,
+ (unsigned long)srcfn);
+ WARN_ON(1);
+ return 0;
+ }
+
+ /*
+ * REVISIT: We should probably pass in the function's size also,
+ * so we can verify that the address to patch exists within
+ * the function
+ */
+ if (sram_addr > omap_sram_base + omap_sram_size ||
+ sram_addr < omap_sram_base + SRAM_BOOTLOADER_SZ) {
+ printk(KERN_ERR "sram: invalid patch address 0x%0lx\n",
+ sram_addr);
+ WARN_ON(1);
+ return 0;
+ }
+
+ if (*(typeof(SRAM_VA_MAGIC) *)sram_addr != SRAM_VA_MAGIC) {
+ printk(KERN_ERR "sram: will not patch address 0x%0lx: "
+ "no magic\n", sram_addr);
+ WARN_ON(1);
+ return 0;
+ }
+#endif /* CONFIG_OMAP_DEBUG_SRAM_PATCH */
+
+ pr_debug("sram: patching 0x%0lx with 0x%0lx\n", sram_addr,
+ (unsigned long)d);
+
+ *(unsigned long *)sram_addr = (unsigned long)d;
+
+ return 1;
+}
+
+
static void omap_sram_error(void)
{
panic("Uninitialized SRAM function\n");
#ifndef __ARCH_ARM_OMAP_SRAM_H
#define __ARCH_ARM_OMAP_SRAM_H
+#include <linux/poison.h> /* for SRAM_VA_MAGIC */
+
extern void * omap_sram_push(void * start, unsigned long size);
+extern int omap_sram_patch_va(void *srcfn, void *srcd, void *sramfn, void __iomem *d);
extern void omap_sram_reprogram_clock(u32 dpllctl, u32 ckctl);
extern void omap2_sram_ddr_init(u32 *slow_dll_ctrl, u32 fast_dll_ctrl,