blob: 103abc13d6234764469f33545f6c1710ae74b55e [file] [log] [blame]
Ralf Baechle23fbee92005-07-25 22:45:45 +00001/*
Atsushi Nemoto22b1d702008-07-11 00:31:36 +09002 * spi_eeprom.c
Ralf Baechle23fbee92005-07-25 22:45:45 +00003 * Copyright (C) 2000-2001 Toshiba Corporation
4 *
5 * 2003-2005 (c) MontaVista Software, Inc. This file is licensed under the
6 * terms of the GNU General Public License version 2. This program is
7 * licensed "as is" without any warranty of any kind, whether express
8 * or implied.
9 *
10 * Support for TX4938 in 2.6 - Manish Lachwani (mlachwani@mvista.com)
11 */
Ralf Baechle23fbee92005-07-25 22:45:45 +000012#include <linux/init.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090013#include <linux/slab.h>
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090014#include <linux/device.h>
15#include <linux/spi/spi.h>
16#include <linux/spi/eeprom.h>
Atsushi Nemoto22b1d702008-07-11 00:31:36 +090017#include <asm/txx9/spi.h>
Ralf Baechle23fbee92005-07-25 22:45:45 +000018
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090019#define AT250X0_PAGE_SIZE 8
Ralf Baechle23fbee92005-07-25 22:45:45 +000020
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090021/* register board information for at25 driver */
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090022int __init spi_eeprom_register(int busid, int chipid, int size)
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090023{
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090024 struct spi_board_info info = {
25 .modalias = "at25",
26 .max_speed_hz = 1500000, /* 1.5Mbps */
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090027 .bus_num = busid,
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090028 .chip_select = chipid,
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090029 /* Mode 0: High-Active, Sample-Then-Shift */
30 };
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090031 struct spi_eeprom *eeprom;
32 eeprom = kzalloc(sizeof(*eeprom), GFP_KERNEL);
33 if (!eeprom)
34 return -ENOMEM;
35 strcpy(eeprom->name, "at250x0");
36 eeprom->byte_len = size;
37 eeprom->page_size = AT250X0_PAGE_SIZE;
38 eeprom->flags = EE_ADDR1;
39 info.platform_data = eeprom;
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090040 return spi_register_board_info(&info, 1);
41}
Ralf Baechle23fbee92005-07-25 22:45:45 +000042
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090043/* simple temporary spi driver to provide early access to seeprom. */
Ralf Baechle23fbee92005-07-25 22:45:45 +000044
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090045static struct read_param {
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090046 int busid;
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090047 int chipid;
48 int address;
49 unsigned char *buf;
50 int len;
51} *read_param;
52
53static int __init early_seeprom_probe(struct spi_device *spi)
54{
55 int stat = 0;
56 u8 cmd[2];
57 int len = read_param->len;
58 char *buf = read_param->buf;
59 int address = read_param->address;
60
61 dev_info(&spi->dev, "spiclk %u KHz.\n",
62 (spi->max_speed_hz + 500) / 1000);
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090063 if (read_param->busid != spi->master->bus_num ||
64 read_param->chipid != spi->chip_select)
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090065 return -ENODEV;
66 while (len > 0) {
67 /* spi_write_then_read can only work with small chunk */
68 int c = len < AT250X0_PAGE_SIZE ? len : AT250X0_PAGE_SIZE;
69 cmd[0] = 0x03; /* AT25_READ */
70 cmd[1] = address;
71 stat = spi_write_then_read(spi, cmd, sizeof(cmd), buf, c);
72 buf += c;
73 len -= c;
74 address += c;
75 }
76 return stat;
77}
78
79static struct spi_driver early_seeprom_driver __initdata = {
80 .driver = {
81 .name = "at25",
82 .owner = THIS_MODULE,
83 },
84 .probe = early_seeprom_probe,
Ralf Baechle23fbee92005-07-25 22:45:45 +000085};
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090086
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090087int __init spi_eeprom_read(int busid, int chipid, int address,
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090088 unsigned char *buf, int len)
Ralf Baechle23fbee92005-07-25 22:45:45 +000089{
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090090 int ret;
91 struct read_param param = {
Atsushi Nemotod75a40e2008-08-19 22:55:14 +090092 .busid = busid,
Atsushi Nemotof74cf6f2007-06-22 23:22:06 +090093 .chipid = chipid,
94 .address = address,
95 .buf = buf,
96 .len = len
97 };
98
99 read_param = &param;
100 ret = spi_register_driver(&early_seeprom_driver);
101 if (!ret)
102 spi_unregister_driver(&early_seeprom_driver);
103 return ret;
Ralf Baechle23fbee92005-07-25 22:45:45 +0000104}