// SPDX-License-Identifier: GPL-2.0-only 
 | 
/* 
 | 
 *  Cobalt NOR flash functions 
 | 
 * 
 | 
 *  Copyright 2012-2015 Cisco Systems, Inc. and/or its affiliates. 
 | 
 *  All rights reserved. 
 | 
 */ 
 | 
  
 | 
#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; 
 | 
  
 | 
        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); 
 | 
    } 
 | 
} 
 |