1*0d6140beSAndroid Build Coastguard Worker /*
2*0d6140beSAndroid Build Coastguard Worker * This file is part of the flashrom project.
3*0d6140beSAndroid Build Coastguard Worker *
4*0d6140beSAndroid Build Coastguard Worker * Copyright (C) 2019 Konstantin Grudnev
5*0d6140beSAndroid Build Coastguard Worker * Copyright (C) 2019 Nikolay Nikolaev
6*0d6140beSAndroid Build Coastguard Worker *
7*0d6140beSAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or modify
8*0d6140beSAndroid Build Coastguard Worker * it under the terms of the GNU General Public License as published by
9*0d6140beSAndroid Build Coastguard Worker * the Free Software Foundation; either version 2 of the License,
10*0d6140beSAndroid Build Coastguard Worker * or any later version.
11*0d6140beSAndroid Build Coastguard Worker * This program is distributed in the hope that it will be useful,
12*0d6140beSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*0d6140beSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14*0d6140beSAndroid Build Coastguard Worker * GNU General Public License for more details.
15*0d6140beSAndroid Build Coastguard Worker */
16*0d6140beSAndroid Build Coastguard Worker
17*0d6140beSAndroid Build Coastguard Worker /*
18*0d6140beSAndroid Build Coastguard Worker * Contains SPI chip driver functions related to ST95XXX series (SPI EEPROM)
19*0d6140beSAndroid Build Coastguard Worker */
20*0d6140beSAndroid Build Coastguard Worker #include <string.h>
21*0d6140beSAndroid Build Coastguard Worker #include <stdlib.h>
22*0d6140beSAndroid Build Coastguard Worker #include "flashchips.h"
23*0d6140beSAndroid Build Coastguard Worker #include "chipdrivers.h"
24*0d6140beSAndroid Build Coastguard Worker #include "spi.h"
25*0d6140beSAndroid Build Coastguard Worker
26*0d6140beSAndroid Build Coastguard Worker /* For ST95XXX chips which have RDID */
probe_spi_st95(struct flashctx * flash)27*0d6140beSAndroid Build Coastguard Worker int probe_spi_st95(struct flashctx *flash)
28*0d6140beSAndroid Build Coastguard Worker {
29*0d6140beSAndroid Build Coastguard Worker /*
30*0d6140beSAndroid Build Coastguard Worker * ST_M95_RDID_OUTSIZE depends on size of the flash and
31*0d6140beSAndroid Build Coastguard Worker * not all ST_M95XXX have RDID.
32*0d6140beSAndroid Build Coastguard Worker */
33*0d6140beSAndroid Build Coastguard Worker static const unsigned char cmd[ST_M95_RDID_OUTSIZE_MAX] = { ST_M95_RDID };
34*0d6140beSAndroid Build Coastguard Worker unsigned char readarr[ST_M95_RDID_INSIZE];
35*0d6140beSAndroid Build Coastguard Worker uint32_t id1, id2;
36*0d6140beSAndroid Build Coastguard Worker int ret;
37*0d6140beSAndroid Build Coastguard Worker
38*0d6140beSAndroid Build Coastguard Worker uint32_t rdid_outsize = ST_M95_RDID_2BA_OUTSIZE; // 16 bit address
39*0d6140beSAndroid Build Coastguard Worker if (flash->chip->total_size * KiB > 64 * KiB)
40*0d6140beSAndroid Build Coastguard Worker rdid_outsize = ST_M95_RDID_3BA_OUTSIZE; // 24 bit address
41*0d6140beSAndroid Build Coastguard Worker
42*0d6140beSAndroid Build Coastguard Worker ret = spi_send_command(flash, rdid_outsize, sizeof(readarr), cmd, readarr);
43*0d6140beSAndroid Build Coastguard Worker if (ret)
44*0d6140beSAndroid Build Coastguard Worker return ret;
45*0d6140beSAndroid Build Coastguard Worker
46*0d6140beSAndroid Build Coastguard Worker id1 = readarr[0]; // manufacture id
47*0d6140beSAndroid Build Coastguard Worker id2 = (readarr[1] << 8) | readarr[2]; // SPI family code + model id
48*0d6140beSAndroid Build Coastguard Worker
49*0d6140beSAndroid Build Coastguard Worker msg_cdbg("%s: id1 0x%02"PRIx32", id2 0x%02"PRIx32"\n", __func__, id1, id2);
50*0d6140beSAndroid Build Coastguard Worker
51*0d6140beSAndroid Build Coastguard Worker if (id1 == flash->chip->manufacture_id && id2 == flash->chip->model_id)
52*0d6140beSAndroid Build Coastguard Worker return 1;
53*0d6140beSAndroid Build Coastguard Worker
54*0d6140beSAndroid Build Coastguard Worker return 0;
55*0d6140beSAndroid Build Coastguard Worker }
56*0d6140beSAndroid Build Coastguard Worker
57*0d6140beSAndroid Build Coastguard Worker /* ST95XXX chips don't have erase operation and erase is made as part of write command */
spi_block_erase_emulation(struct flashctx * flash,unsigned int addr,unsigned int blocklen)58*0d6140beSAndroid Build Coastguard Worker int spi_block_erase_emulation(struct flashctx *flash, unsigned int addr, unsigned int blocklen)
59*0d6140beSAndroid Build Coastguard Worker {
60*0d6140beSAndroid Build Coastguard Worker uint8_t *erased_contents = NULL;
61*0d6140beSAndroid Build Coastguard Worker int result = 0;
62*0d6140beSAndroid Build Coastguard Worker
63*0d6140beSAndroid Build Coastguard Worker erased_contents = (uint8_t *)malloc(blocklen * sizeof(uint8_t));
64*0d6140beSAndroid Build Coastguard Worker if (!erased_contents) {
65*0d6140beSAndroid Build Coastguard Worker msg_cerr("Out of memory!\n");
66*0d6140beSAndroid Build Coastguard Worker return 1;
67*0d6140beSAndroid Build Coastguard Worker }
68*0d6140beSAndroid Build Coastguard Worker memset(erased_contents, ERASED_VALUE(flash), blocklen * sizeof(uint8_t));
69*0d6140beSAndroid Build Coastguard Worker result = spi_write_chunked(flash, erased_contents, 0, blocklen, flash->chip->page_size);
70*0d6140beSAndroid Build Coastguard Worker free(erased_contents);
71*0d6140beSAndroid Build Coastguard Worker return result;
72*0d6140beSAndroid Build Coastguard Worker }
73