blob: cc40b74940f5f6531e4e74cfcf4c665a7b13a9fd [file] [log] [blame]
Waldemar Brodkorb121915c2010-06-08 19:06:01 +02001/*
2 * BCM947xx nvram variable access
3 *
4 * Copyright (C) 2005 Broadcom Corporation
5 * Copyright (C) 2006 Felix Fietkau <nbd@openwrt.org>
Hauke Mehrtensf36738d2012-12-26 19:51:13 +00006 * Copyright (C) 2010-2012 Hauke Mehrtens <hauke@hauke-m.de>
Waldemar Brodkorb121915c2010-06-08 19:06:01 +02007 *
Ralf Baechle70342282013-01-22 12:59:30 +01008 * This program is free software; you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020010 * Free Software Foundation; either version 2 of the License, or (at your
11 * option) any later version.
12 */
13
14#include <linux/init.h>
15#include <linux/types.h>
16#include <linux/module.h>
17#include <linux/ssb/ssb.h>
18#include <linux/kernel.h>
19#include <linux/string.h>
20#include <asm/addrspace.h>
Hauke Mehrtens111bd982012-12-26 19:51:14 +000021#include <bcm47xx_nvram.h>
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020022#include <asm/mach-bcm47xx/bcm47xx.h>
23
24static char nvram_buf[NVRAM_SPACE];
25
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000026static u32 find_nvram_size(u32 end)
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020027{
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020028 struct nvram_header *header;
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000029 u32 nvram_sizes[] = {0x8000, 0xF000, 0x10000};
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020030 int i;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020031
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000032 for (i = 0; i < ARRAY_SIZE(nvram_sizes); i++) {
33 header = (struct nvram_header *)KSEG1ADDR(end - nvram_sizes[i]);
34 if (header->magic == NVRAM_HEADER)
35 return nvram_sizes[i];
Hauke Mehrtens08ccf5722011-07-23 01:20:12 +020036 }
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020037
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000038 return 0;
39}
40
41/* Probe for NVRAM header */
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +000042static int nvram_find_and_copy(u32 base, u32 lim)
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020043{
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020044 struct nvram_header *header;
45 int i;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020046 u32 off;
47 u32 *src, *dst;
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000048 u32 size;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020049
Hauke Mehrtensc4485672012-12-26 19:51:11 +000050 /* TODO: when nvram is on nand flash check for bad blocks first. */
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020051 off = FLASH_MIN;
52 while (off <= lim) {
53 /* Windowed flash access */
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000054 size = find_nvram_size(base + off);
55 if (size) {
56 header = (struct nvram_header *)KSEG1ADDR(base + off -
57 size);
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020058 goto found;
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000059 }
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020060 off <<= 1;
61 }
62
63 /* Try embedded NVRAM at 4 KB and 1 KB as last resorts */
64 header = (struct nvram_header *) KSEG1ADDR(base + 4096);
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000065 if (header->magic == NVRAM_HEADER) {
66 size = NVRAM_SPACE;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020067 goto found;
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000068 }
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020069
70 header = (struct nvram_header *) KSEG1ADDR(base + 1024);
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000071 if (header->magic == NVRAM_HEADER) {
72 size = NVRAM_SPACE;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020073 goto found;
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000074 }
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020075
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000076 pr_err("no nvram found\n");
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +000077 return -ENXIO;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020078
79found:
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000080
81 if (header->len > size)
82 pr_err("The nvram size accoridng to the header seems to be bigger than the partition on flash\n");
83 if (header->len > NVRAM_SPACE)
84 pr_err("nvram on flash (%i bytes) is bigger than the reserved space in memory, will just copy the first %i bytes\n",
85 header->len, NVRAM_SPACE);
86
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020087 src = (u32 *) header;
88 dst = (u32 *) nvram_buf;
89 for (i = 0; i < sizeof(struct nvram_header); i += 4)
90 *dst++ = *src++;
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000091 for (; i < header->len && i < NVRAM_SPACE && i < size; i += 4)
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020092 *dst++ = le32_to_cpu(*src++);
Hauke Mehrtensf36738d2012-12-26 19:51:13 +000093 memset(dst, 0x0, NVRAM_SPACE - i);
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +000094
95 return 0;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +020096}
97
Rafał Miłeckibb765632012-12-26 08:29:17 +000098#ifdef CONFIG_BCM47XX_SSB
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +000099static int nvram_init_ssb(void)
Rafał Miłeckibb765632012-12-26 08:29:17 +0000100{
101 struct ssb_mipscore *mcore = &bcm47xx_bus.ssb.mipscore;
102 u32 base;
103 u32 lim;
104
105 if (mcore->pflash.present) {
106 base = mcore->pflash.window;
107 lim = mcore->pflash.window_size;
108 } else {
109 pr_err("Couldn't find supported flash memory\n");
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000110 return -ENXIO;
Rafał Miłeckibb765632012-12-26 08:29:17 +0000111 }
112
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000113 return nvram_find_and_copy(base, lim);
Rafał Miłeckibb765632012-12-26 08:29:17 +0000114}
115#endif
116
117#ifdef CONFIG_BCM47XX_BCMA
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000118static int nvram_init_bcma(void)
Rafał Miłeckibb765632012-12-26 08:29:17 +0000119{
120 struct bcma_drv_cc *cc = &bcm47xx_bus.bcma.bus.drv_cc;
121 u32 base;
122 u32 lim;
123
Hauke Mehrtensc4485672012-12-26 19:51:11 +0000124#ifdef CONFIG_BCMA_NFLASH
125 if (cc->nflash.boot) {
126 base = BCMA_SOC_FLASH1;
127 lim = BCMA_SOC_FLASH1_SZ;
128 } else
129#endif
Rafał Miłeckibb765632012-12-26 08:29:17 +0000130 if (cc->pflash.present) {
131 base = cc->pflash.window;
132 lim = cc->pflash.window_size;
133#ifdef CONFIG_BCMA_SFLASH
134 } else if (cc->sflash.present) {
135 base = cc->sflash.window;
136 lim = cc->sflash.size;
137#endif
138 } else {
139 pr_err("Couldn't find supported flash memory\n");
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000140 return -ENXIO;
Rafał Miłeckibb765632012-12-26 08:29:17 +0000141 }
142
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000143 return nvram_find_and_copy(base, lim);
Rafał Miłeckibb765632012-12-26 08:29:17 +0000144}
145#endif
146
Hauke Mehrtense58da162012-12-26 19:51:12 +0000147static int nvram_init(void)
Rafał Miłeckibb765632012-12-26 08:29:17 +0000148{
149 switch (bcm47xx_bus_type) {
150#ifdef CONFIG_BCM47XX_SSB
151 case BCM47XX_BUS_TYPE_SSB:
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000152 return nvram_init_ssb();
Rafał Miłeckibb765632012-12-26 08:29:17 +0000153#endif
154#ifdef CONFIG_BCM47XX_BCMA
155 case BCM47XX_BUS_TYPE_BCMA:
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000156 return nvram_init_bcma();
Rafał Miłeckibb765632012-12-26 08:29:17 +0000157#endif
158 }
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000159 return -ENXIO;
Rafał Miłeckibb765632012-12-26 08:29:17 +0000160}
161
Hauke Mehrtens111bd982012-12-26 19:51:14 +0000162int bcm47xx_nvram_getenv(char *name, char *val, size_t val_len)
Waldemar Brodkorb121915c2010-06-08 19:06:01 +0200163{
164 char *var, *value, *end, *eq;
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000165 int err;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +0200166
167 if (!name)
Hauke Mehrtensee7e2f32012-12-26 19:51:09 +0000168 return -EINVAL;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +0200169
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000170 if (!nvram_buf[0]) {
Hauke Mehrtense58da162012-12-26 19:51:12 +0000171 err = nvram_init();
Hauke Mehrtenscc4403e2012-12-26 19:51:10 +0000172 if (err)
173 return err;
174 }
Waldemar Brodkorb121915c2010-06-08 19:06:01 +0200175
176 /* Look for name=value and return value */
177 var = &nvram_buf[sizeof(struct nvram_header)];
178 end = nvram_buf + sizeof(nvram_buf) - 2;
179 end[0] = end[1] = '\0';
180 for (; *var; var = value + strlen(value) + 1) {
181 eq = strchr(var, '=');
182 if (!eq)
183 break;
184 value = eq + 1;
185 if ((eq - var) == strlen(name) &&
186 strncmp(var, name, (eq - var)) == 0) {
Hauke Mehrtens44d4b2a2012-02-28 00:56:11 +0100187 return snprintf(val, val_len, "%s", value);
Waldemar Brodkorb121915c2010-06-08 19:06:01 +0200188 }
189 }
Hauke Mehrtensee7e2f32012-12-26 19:51:09 +0000190 return -ENOENT;
Waldemar Brodkorb121915c2010-06-08 19:06:01 +0200191}
Hauke Mehrtens111bd982012-12-26 19:51:14 +0000192EXPORT_SYMBOL(bcm47xx_nvram_getenv);