hc
2023-02-13 e440ec23c5a540cdd3f7464e8779219be6fd3d95
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
/*
 * spi_eeprom.c
 * Copyright (C) 2000-2001 Toshiba Corporation
 *
 * 2003-2005 (c) MontaVista Software, Inc. This file is licensed under the
 * terms of the GNU General Public License version 2. This program is
 * licensed "as is" without any warranty of any kind, whether express
 * or implied.
 *
 * Support for TX4938 in 2.6 - Manish Lachwani (mlachwani@mvista.com)
 */
#include <linux/init.h>
#include <linux/slab.h>
#include <linux/export.h>
#include <linux/device.h>
#include <linux/spi/spi.h>
#include <linux/spi/eeprom.h>
#include <asm/txx9/spi.h>
 
#define AT250X0_PAGE_SIZE    8
 
/* register board information for at25 driver */
int __init spi_eeprom_register(int busid, int chipid, int size)
{
   struct spi_board_info info = {
       .modalias = "at25",
       .max_speed_hz = 1500000,    /* 1.5Mbps */
       .bus_num = busid,
       .chip_select = chipid,
       /* Mode 0: High-Active, Sample-Then-Shift */
   };
   struct spi_eeprom *eeprom;
   eeprom = kzalloc(sizeof(*eeprom), GFP_KERNEL);
   if (!eeprom)
       return -ENOMEM;
   strcpy(eeprom->name, "at250x0");
   eeprom->byte_len = size;
   eeprom->page_size = AT250X0_PAGE_SIZE;
   eeprom->flags = EE_ADDR1;
   info.platform_data = eeprom;
   return spi_register_board_info(&info, 1);
}
 
/* simple temporary spi driver to provide early access to seeprom. */
 
static struct read_param {
   int busid;
   int chipid;
   int address;
   unsigned char *buf;
   int len;
} *read_param;
 
static int __init early_seeprom_probe(struct spi_device *spi)
{
   int stat = 0;
   u8 cmd[2];
   int len = read_param->len;
   char *buf = read_param->buf;
   int address = read_param->address;
 
   dev_info(&spi->dev, "spiclk %u KHz.\n",
        (spi->max_speed_hz + 500) / 1000);
   if (read_param->busid != spi->master->bus_num ||
       read_param->chipid != spi->chip_select)
       return -ENODEV;
   while (len > 0) {
       /* spi_write_then_read can only work with small chunk */
       int c = len < AT250X0_PAGE_SIZE ? len : AT250X0_PAGE_SIZE;
       cmd[0] = 0x03;    /* AT25_READ */
       cmd[1] = address;
       stat = spi_write_then_read(spi, cmd, sizeof(cmd), buf, c);
       buf += c;
       len -= c;
       address += c;
   }
   return stat;
}
 
static struct spi_driver early_seeprom_driver __initdata = {
   .driver = {
       .name    = "at25",
   },
   .probe    = early_seeprom_probe,
};
 
int __init spi_eeprom_read(int busid, int chipid, int address,
              unsigned char *buf, int len)
{
   int ret;
   struct read_param param = {
       .busid = busid,
       .chipid = chipid,
       .address = address,
       .buf = buf,
       .len = len
   };
 
   read_param = &param;
   ret = spi_register_driver(&early_seeprom_driver);
   if (!ret)
       spi_unregister_driver(&early_seeprom_driver);
   return ret;
}