123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129 |
- /*
- * Cobalt NOR flash functions
- *
- * Copyright 2012-2015 Cisco Systems, Inc. and/or its affiliates.
- * All rights reserved.
- *
- * This program is free software; you may redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; version 2 of the License.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
- * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
- * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
- * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- * SOFTWARE.
- */
- #include <linux/mtd/mtd.h>
- #include <linux/mtd/map.h>
- #include <linux/mtd/cfi.h>
- #include <linux/time.h>
- #include "cobalt-flash.h"
- #define ADRS(offset) (COBALT_BUS_FLASH_BASE + offset)
- static struct map_info cobalt_flash_map = {
- .name = "cobalt-flash",
- .bankwidth = 2, /* 16 bits */
- .size = 0x4000000, /* 64MB */
- .phys = 0, /* offset */
- };
- static map_word flash_read16(struct map_info *map, unsigned long offset)
- {
- map_word r;
- r.x[0] = cobalt_bus_read32(map->virt, ADRS(offset));
- if (offset & 0x2)
- r.x[0] >>= 16;
- else
- r.x[0] &= 0x0000ffff;
- return r;
- }
- static void flash_write16(struct map_info *map, const map_word datum,
- unsigned long offset)
- {
- u16 data = (u16)datum.x[0];
- cobalt_bus_write16(map->virt, ADRS(offset), data);
- }
- static void flash_copy_from(struct map_info *map, void *to,
- unsigned long from, ssize_t len)
- {
- u32 src = from;
- u8 *dest = to;
- u32 data;
- while (len) {
- data = cobalt_bus_read32(map->virt, ADRS(src));
- do {
- *dest = data >> (8 * (src & 3));
- src++;
- dest++;
- len--;
- } while (len && (src % 4));
- }
- }
- static void flash_copy_to(struct map_info *map, unsigned long to,
- const void *from, ssize_t len)
- {
- const u8 *src = from;
- u32 dest = to;
- pr_info("%s: offset 0x%x: length %zu\n", __func__, dest, len);
- while (len) {
- u16 data = 0xffff;
- do {
- data = *src << (8 * (dest & 1));
- src++;
- dest++;
- len--;
- } while (len && (dest % 2));
- cobalt_bus_write16(map->virt, ADRS(dest - 2), data);
- }
- }
- int cobalt_flash_probe(struct cobalt *cobalt)
- {
- struct map_info *map = &cobalt_flash_map;
- struct mtd_info *mtd;
- BUG_ON(!map_bankwidth_supported(map->bankwidth));
- map->virt = cobalt->bar1;
- map->read = flash_read16;
- map->write = flash_write16;
- map->copy_from = flash_copy_from;
- map->copy_to = flash_copy_to;
- mtd = do_map_probe("cfi_probe", map);
- cobalt->mtd = mtd;
- if (!mtd) {
- cobalt_err("Probe CFI flash failed!\n");
- return -1;
- }
- mtd->owner = THIS_MODULE;
- mtd->dev.parent = &cobalt->pci_dev->dev;
- mtd_device_register(mtd, NULL, 0);
- return 0;
- }
- void cobalt_flash_remove(struct cobalt *cobalt)
- {
- if (cobalt->mtd) {
- mtd_device_unregister(cobalt->mtd);
- map_destroy(cobalt->mtd);
- }
- }
|