mirror of https://github.com/ipxe/ipxe.git
Abstracted out part of the concept of an SPI device to a generalised NVS
device. Separated the mechanisms of non-volatile storage access and non-volatile stored options.pull/1/head
parent
dc06c895fc
commit
946967f09c
|
@ -0,0 +1,109 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2 of the
|
||||||
|
* License, or any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <gpxe/dhcp.h>
|
||||||
|
#include <gpxe/nvs.h>
|
||||||
|
#include <gpxe/nvo.h>
|
||||||
|
|
||||||
|
/** @file
|
||||||
|
*
|
||||||
|
* Non-volatile stored options
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
static size_t nvo_options_len ( struct nvs_options *nvo ) {
|
||||||
|
struct dhcp_option *option;
|
||||||
|
uint8_t sum;
|
||||||
|
unsigned int i;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
for ( sum = 0, i = 0 ; i < nvo->nvs->size ; i++ ) {
|
||||||
|
sum += * ( ( uint8_t * ) ( nvo->options->data + i ) );
|
||||||
|
}
|
||||||
|
if ( sum != 0 ) {
|
||||||
|
DBG ( "NVO %p has bad checksum %02x; assuming empty\n",
|
||||||
|
nvo, sum );
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
option = nvo->options->data;
|
||||||
|
if ( option->tag == DHCP_PAD ) {
|
||||||
|
DBG ( "NVO %p has bad start; assuming empty\n", nvo );
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
option = find_dhcp_option ( nvo->options, DHCP_END );
|
||||||
|
if ( ! option ) {
|
||||||
|
DBG ( "NVO %p has no end tag; assuming empty\n", nvo );
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
len = ( ( void * ) option - nvo->options->data + 1 );
|
||||||
|
DBG ( "NVO %p contains %zd bytes of options (maximum %zd)\n",
|
||||||
|
nvo, len, nvo->nvs->size );
|
||||||
|
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
int nvo_register ( struct nvs_options *nvo ) {
|
||||||
|
struct dhcp_option *option;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
nvo->options = alloc_dhcp_options ( nvo->nvs->size );
|
||||||
|
if ( ! nvo->options ) {
|
||||||
|
DBG ( "NVO %p could not allocate %zd bytes\n",
|
||||||
|
nvo, nvo->nvs->size );
|
||||||
|
rc = -ENOMEM;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ( rc = nvo->nvs->read ( nvo->nvs, 0, nvo->options->data,
|
||||||
|
nvo->nvs->size ) ) != 0 ) {
|
||||||
|
DBG ( "NVO %p could not read [0,%zd)\n",
|
||||||
|
nvo, nvo->nvs->size );
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
nvo->options->len = nvo->options->max_len;
|
||||||
|
nvo->options->len = nvo_options_len ( nvo );
|
||||||
|
if ( ! nvo->options->len ) {
|
||||||
|
option = nvo->options->data;
|
||||||
|
option->tag = DHCP_END;
|
||||||
|
nvo->options->len = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
register_dhcp_options ( nvo->options );
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err:
|
||||||
|
|
||||||
|
free_dhcp_options ( nvo->options );
|
||||||
|
nvo->options = NULL;
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
void nvo_unregister ( struct nvs_options *nvo ) {
|
||||||
|
if ( nvo->options ) {
|
||||||
|
unregister_dhcp_options ( nvo->options );
|
||||||
|
free_dhcp_options ( nvo->options );
|
||||||
|
nvo->options = NULL;
|
||||||
|
}
|
||||||
|
}
|
106
src/core/nvs.c
106
src/core/nvs.c
|
@ -1,106 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>.
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU General Public License as
|
|
||||||
* published by the Free Software Foundation; either version 2 of the
|
|
||||||
* License, or any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful, but
|
|
||||||
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <gpxe/dhcp.h>
|
|
||||||
#include <gpxe/nvs.h>
|
|
||||||
|
|
||||||
/** @file
|
|
||||||
*
|
|
||||||
* Non-volatile storage
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
static size_t nvs_options_len ( struct nvs_device *nvs ) {
|
|
||||||
struct dhcp_option *option;
|
|
||||||
uint8_t sum;
|
|
||||||
unsigned int i;
|
|
||||||
size_t len;
|
|
||||||
|
|
||||||
for ( sum = 0, i = 0 ; i < nvs->len ; i++ ) {
|
|
||||||
sum += * ( ( uint8_t * ) ( nvs->options->data + i ) );
|
|
||||||
}
|
|
||||||
if ( sum != 0 ) {
|
|
||||||
DBG ( "NVS %p has bad checksum %02x; assuming empty\n",
|
|
||||||
nvs, sum );
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
option = nvs->options->data;
|
|
||||||
if ( option->tag == DHCP_PAD ) {
|
|
||||||
DBG ( "NVS %p has bad start; assuming empty\n", nvs );
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
option = find_dhcp_option ( nvs->options, DHCP_END );
|
|
||||||
if ( ! option ) {
|
|
||||||
DBG ( "NVS %p has no end tag; assuming empty\n", nvs );
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
len = ( ( void * ) option - nvs->options->data + 1 );
|
|
||||||
DBG ( "NVS %p contains %zd bytes of options (maximum %zd)\n",
|
|
||||||
nvs, len, nvs->len );
|
|
||||||
|
|
||||||
return len;
|
|
||||||
}
|
|
||||||
|
|
||||||
int nvs_register ( struct nvs_device *nvs ) {
|
|
||||||
struct dhcp_option *option;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
nvs->options = alloc_dhcp_options ( nvs->len );
|
|
||||||
if ( ! nvs->options ) {
|
|
||||||
DBG ( "NVS %p could not allocate %zd bytes\n", nvs, nvs->len );
|
|
||||||
rc = -ENOMEM;
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( ( rc = nvs->op->read ( nvs, 0, nvs->options->data,
|
|
||||||
nvs->len ) ) != 0 ) {
|
|
||||||
DBG ( "NVS %p could not read [0,%zd)\n", nvs, nvs->len );
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
nvs->options->len = nvs->options->max_len;
|
|
||||||
nvs->options->len = nvs_options_len ( nvs );
|
|
||||||
if ( ! nvs->options->len ) {
|
|
||||||
option = nvs->options->data;
|
|
||||||
option->tag = DHCP_END;
|
|
||||||
nvs->options->len = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
register_dhcp_options ( nvs->options );
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
err:
|
|
||||||
|
|
||||||
free_dhcp_options ( nvs->options );
|
|
||||||
nvs->options = NULL;
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
void nvs_unregister ( struct nvs_device *nvs ) {
|
|
||||||
if ( nvs->options ) {
|
|
||||||
unregister_dhcp_options ( nvs->options );
|
|
||||||
free_dhcp_options ( nvs->options );
|
|
||||||
nvs->options = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -147,22 +147,21 @@ static int spi_bit_rw ( struct spi_bus *bus, struct spi_device *device,
|
||||||
const void *data_out, void *data_in, size_t len ) {
|
const void *data_out, void *data_in, size_t len ) {
|
||||||
struct spi_bit_basher *spibit
|
struct spi_bit_basher *spibit
|
||||||
= container_of ( bus, struct spi_bit_basher, bus );
|
= container_of ( bus, struct spi_bit_basher, bus );
|
||||||
struct spi_device_type *devtype = device->type;
|
|
||||||
uint32_t tmp;
|
uint32_t tmp;
|
||||||
|
|
||||||
/* Assert chip select on specified slave */
|
/* Assert chip select on specified slave */
|
||||||
spi_bit_set_slave_select ( spibit, device->slave, SELECT_SLAVE );
|
spi_bit_set_slave_select ( spibit, device->slave, SELECT_SLAVE );
|
||||||
|
|
||||||
/* Transmit command */
|
/* Transmit command */
|
||||||
assert ( devtype->command_len <= ( 8 * sizeof ( tmp ) ) );
|
assert ( device->command_len <= ( 8 * sizeof ( tmp ) ) );
|
||||||
tmp = cpu_to_le32 ( command );
|
tmp = cpu_to_le32 ( command );
|
||||||
spi_bit_transfer ( spibit, &tmp, NULL, devtype->command_len );
|
spi_bit_transfer ( spibit, &tmp, NULL, device->command_len );
|
||||||
|
|
||||||
/* Transmit address, if present */
|
/* Transmit address, if present */
|
||||||
if ( address >= 0 ) {
|
if ( address >= 0 ) {
|
||||||
assert ( devtype->address_len <= ( 8 * sizeof ( tmp ) ) );
|
assert ( device->address_len <= ( 8 * sizeof ( tmp ) ) );
|
||||||
tmp = cpu_to_le32 ( address );
|
tmp = cpu_to_le32 ( address );
|
||||||
spi_bit_transfer ( spibit, &tmp, NULL, devtype->address_len );
|
spi_bit_transfer ( spibit, &tmp, NULL, device->address_len );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Transmit/receive data */
|
/* Transmit/receive data */
|
||||||
|
|
|
@ -2388,11 +2388,6 @@ static int falcon_write_nvs ( struct nvs_device *nvs, unsigned int offset,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct nvs_operations falcon_nvs_operations = {
|
|
||||||
.read = falcon_read_nvs,
|
|
||||||
.write = falcon_write_nvs,
|
|
||||||
};
|
|
||||||
|
|
||||||
/** RX descriptor */
|
/** RX descriptor */
|
||||||
typedef efab_qword_t falcon_rx_desc_t;
|
typedef efab_qword_t falcon_rx_desc_t;
|
||||||
|
|
||||||
|
@ -3046,10 +3041,12 @@ static int falcon_init_nic ( struct efab_nic *efab ) {
|
||||||
|
|
||||||
/* Register non-volatile storage */
|
/* Register non-volatile storage */
|
||||||
if ( efab->has_eeprom ) {
|
if ( efab->has_eeprom ) {
|
||||||
|
/*
|
||||||
efab->nvs.op = &falcon_nvs_operations;
|
efab->nvs.op = &falcon_nvs_operations;
|
||||||
efab->nvs.len = 0x100;
|
efab->nvs.len = 0x100;
|
||||||
if ( nvs_register ( &efab->nvs ) != 0 )
|
if ( nvs_register ( &efab->nvs ) != 0 )
|
||||||
return 0;
|
return 0;
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -240,15 +240,12 @@ static struct bit_basher_operations rtl_basher_ops = {
|
||||||
.write = rtl_spi_write_bit,
|
.write = rtl_spi_write_bit,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct spi_device_type rtl_ee9346 = AT93C46 ( 16 );
|
|
||||||
static struct spi_device_type rtl_ee9356 = AT93C56 ( 16 );
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set up for EEPROM access
|
* Set up for EEPROM access
|
||||||
*
|
*
|
||||||
* @v rtl RTL8139 NIC
|
* @v rtl RTL8139 NIC
|
||||||
*/
|
*/
|
||||||
static void rtl_init_eeprom ( struct rtl8139_nic *rtl ) {
|
void rtl_init_eeprom ( struct rtl8139_nic *rtl ) {
|
||||||
int ee9356;
|
int ee9356;
|
||||||
|
|
||||||
/* Initialise three-wire bus */
|
/* Initialise three-wire bus */
|
||||||
|
@ -258,8 +255,13 @@ static void rtl_init_eeprom ( struct rtl8139_nic *rtl ) {
|
||||||
|
|
||||||
/* Detect EEPROM type and initialise three-wire device */
|
/* Detect EEPROM type and initialise three-wire device */
|
||||||
ee9356 = ( inw ( rtl->ioaddr + RxConfig ) & Eeprom9356 );
|
ee9356 = ( inw ( rtl->ioaddr + RxConfig ) & Eeprom9356 );
|
||||||
DBG ( "EEPROM is an %s\n", ( ee9356 ? "AT93C56" : "AT93C46" ) );
|
if ( ee9356 ) {
|
||||||
rtl->eeprom.type = ( ee9356 ? &rtl_ee9356 : &rtl_ee9346 );
|
DBG ( "EEPROM is an AT93C56\n" );
|
||||||
|
init_at93c56 ( &rtl->eeprom, 16 );
|
||||||
|
} else {
|
||||||
|
DBG ( "EEPROM is an AT93C46\n" );
|
||||||
|
init_at93c46 ( &rtl->eeprom, 16 );
|
||||||
|
}
|
||||||
rtl->eeprom.bus = &rtl->spibit.bus;
|
rtl->eeprom.bus = &rtl->spibit.bus;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -271,12 +273,12 @@ static void rtl_init_eeprom ( struct rtl8139_nic *rtl ) {
|
||||||
*/
|
*/
|
||||||
static void rtl_read_mac ( struct rtl8139_nic *rtl, uint8_t *mac_addr ) {
|
static void rtl_read_mac ( struct rtl8139_nic *rtl, uint8_t *mac_addr ) {
|
||||||
|
|
||||||
struct spi_device *device = &rtl->eeprom;
|
struct nvs_device *nvs = &rtl->eeprom.nvs;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
DBG ( "MAC address is " );
|
DBG ( "MAC address is " );
|
||||||
for ( i = EE_MAC ; i < ( EE_MAC + ( ETH_ALEN / 2 ) ) ; i++ ) {
|
for ( i = EE_MAC ; i < ( EE_MAC + ( ETH_ALEN / 2 ) ) ; i++ ) {
|
||||||
device->type->read ( device, i, mac_addr, 2 );
|
nvs_read ( nvs, i, mac_addr, 2 );
|
||||||
DBG ( "%02x%02x", mac_addr[0], mac_addr[1] );
|
DBG ( "%02x%02x", mac_addr[0], mac_addr[1] );
|
||||||
mac_addr += 2;
|
mac_addr += 2;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2 of the
|
||||||
|
* License, or any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <gpxe/nvs.h>
|
||||||
|
|
||||||
|
/** @file
|
||||||
|
*
|
||||||
|
* Non-volatile storage
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
int nvs_read ( struct nvs_device *nvs, unsigned int address,
|
||||||
|
void *data, size_t len ) {
|
||||||
|
return nvs->read ( nvs, address, data, len );
|
||||||
|
}
|
|
@ -28,14 +28,15 @@
|
||||||
|
|
||||||
/** Read data from three-wire device
|
/** Read data from three-wire device
|
||||||
*
|
*
|
||||||
* @v device SPI device
|
* @v nvs NVS device
|
||||||
* @v address Address from which to read
|
* @v address Address from which to read
|
||||||
* @v data Data buffer
|
* @v data Data buffer
|
||||||
* @v len Length of data buffer
|
* @v len Length of data buffer
|
||||||
* @ret rc Return status code
|
* @ret rc Return status code
|
||||||
*/
|
*/
|
||||||
int threewire_read ( struct spi_device *device, unsigned int address,
|
int threewire_read ( struct nvs_device *nvs, unsigned int address,
|
||||||
void *data, size_t len ) {
|
void *data, size_t len ) {
|
||||||
|
struct spi_device *device = nvs_to_spi ( nvs );
|
||||||
struct spi_bus *bus = device->bus;
|
struct spi_bus *bus = device->bus;
|
||||||
|
|
||||||
assert ( bus->mode == SPI_MODE_THREEWIRE );
|
assert ( bus->mode == SPI_MODE_THREEWIRE );
|
||||||
|
|
|
@ -0,0 +1,21 @@
|
||||||
|
#ifndef _GPXE_NVO_H
|
||||||
|
#define _GPXE_NVO_H
|
||||||
|
|
||||||
|
/** @file
|
||||||
|
*
|
||||||
|
* Non-volatile stored options
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct nvs_device;
|
||||||
|
struct dhcp_option_block;
|
||||||
|
|
||||||
|
struct nvs_options {
|
||||||
|
struct nvs_device *nvs;
|
||||||
|
struct dhcp_option_block *options;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern int nvo_register ( struct nvs_options *nvo );
|
||||||
|
extern void nvo_unregister ( struct nvs_options *nvo );
|
||||||
|
|
||||||
|
#endif /* _GPXE_NVO_H */
|
|
@ -9,22 +9,53 @@
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
struct nvs_operations;
|
/** A non-volatile storage device */
|
||||||
|
|
||||||
struct nvs_device {
|
struct nvs_device {
|
||||||
struct dhcp_option_block *options;
|
/** Word length, in bits */
|
||||||
size_t len;
|
unsigned int word_len;
|
||||||
struct nvs_operations *op;
|
/** Device size (in words) */
|
||||||
};
|
unsigned int size;
|
||||||
|
/** Data block size (in words)
|
||||||
struct nvs_operations {
|
*
|
||||||
int ( * read ) ( struct nvs_device *nvs, unsigned int offset,
|
* This is the block size used by the device. It must be a
|
||||||
|
* power of two. Data reads and writes must not cross a block
|
||||||
|
* boundary.
|
||||||
|
*
|
||||||
|
* Many devices allow reads to cross a block boundary, and
|
||||||
|
* restrict only writes. For the sake of simplicity, we
|
||||||
|
* assume that the same restriction applies to both reads and
|
||||||
|
* writes.
|
||||||
|
*/
|
||||||
|
unsigned int block_size;
|
||||||
|
/** Read data from device
|
||||||
|
*
|
||||||
|
* @v nvs NVS device
|
||||||
|
* @v address Address from which to read
|
||||||
|
* @v data Data buffer
|
||||||
|
* @v len Length of data buffer
|
||||||
|
* @ret rc Return status code
|
||||||
|
*
|
||||||
|
* Reads may not cross a block boundary.
|
||||||
|
*/
|
||||||
|
int ( * read ) ( struct nvs_device *nvs, unsigned int address,
|
||||||
void *data, size_t len );
|
void *data, size_t len );
|
||||||
int ( * write ) ( struct nvs_device *nvs, unsigned int offset,
|
/** Write data to device
|
||||||
|
*
|
||||||
|
* @v nvs NVS device
|
||||||
|
* @v address Address to which to write
|
||||||
|
* @v data Data buffer
|
||||||
|
* @v len Length of data buffer
|
||||||
|
* @ret rc Return status code
|
||||||
|
*
|
||||||
|
* Writes may not cross a block boundary.
|
||||||
|
*/
|
||||||
|
int ( * write ) ( struct nvs_device *nvs, unsigned int address,
|
||||||
const void *data, size_t len );
|
const void *data, size_t len );
|
||||||
};
|
};
|
||||||
|
|
||||||
extern int nvs_register ( struct nvs_device *nvs );
|
extern int nvs_read ( struct nvs_device *nvs, unsigned int address,
|
||||||
extern void nvs_unregister ( struct nvs_device *nvs );
|
void *data, size_t len );
|
||||||
|
extern int nvs_write ( struct nvs_device *nvs, unsigned int address,
|
||||||
|
const void *data, size_t len );
|
||||||
|
|
||||||
#endif /* _GPXE_NVS_H */
|
#endif /* _GPXE_NVS_H */
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <gpxe/bitbash.h>
|
#include <gpxe/nvs.h>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @defgroup spicmds SPI commands
|
* @defgroup spicmds SPI commands
|
||||||
|
@ -75,32 +75,19 @@
|
||||||
|
|
||||||
/** @} */
|
/** @} */
|
||||||
|
|
||||||
struct spi_device;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An SPI device type
|
* An SPI device
|
||||||
*
|
*
|
||||||
* This data structure represents all the characteristics belonging to
|
* This data structure represents a physical SPI device attached to an
|
||||||
* a particular type of SPI device, e.g. "an Atmel 251024 serial flash",
|
* SPI bus.
|
||||||
* or "a Microchip 25040 serial EEPROM".
|
|
||||||
*/
|
*/
|
||||||
struct spi_device_type {
|
struct spi_device {
|
||||||
/** Word length, in bits */
|
/** NVS device */
|
||||||
unsigned int word_len;
|
struct nvs_device nvs;
|
||||||
/** Device size (in words) */
|
/** SPI bus to which device is attached */
|
||||||
unsigned int size;
|
struct spi_bus *bus;
|
||||||
/** Data block size (in words)
|
/** Slave number */
|
||||||
*
|
unsigned int slave;
|
||||||
* This is the block size used by the device. It must be a
|
|
||||||
* power of two. Data reads and writes must not cross a block
|
|
||||||
* boundary.
|
|
||||||
*
|
|
||||||
* Many devices allow reads to cross a block boundary, and
|
|
||||||
* restrict only writes. For the sake of simplicity, we
|
|
||||||
* assume that the same restriction applies to both reads and
|
|
||||||
* writes.
|
|
||||||
*/
|
|
||||||
unsigned int block_size;
|
|
||||||
/** Command length, in bits */
|
/** Command length, in bits */
|
||||||
unsigned int command_len;
|
unsigned int command_len;
|
||||||
/** Address length, in bits */
|
/** Address length, in bits */
|
||||||
|
@ -113,64 +100,18 @@ struct spi_device_type {
|
||||||
* commands should be munged in this way.
|
* commands should be munged in this way.
|
||||||
*/
|
*/
|
||||||
unsigned int munge_address : 1;
|
unsigned int munge_address : 1;
|
||||||
/** Read data from device
|
|
||||||
*
|
|
||||||
* @v device SPI device
|
|
||||||
* @v address Address from which to read
|
|
||||||
* @v data Data buffer
|
|
||||||
* @v len Length of data buffer
|
|
||||||
* @ret rc Return status code
|
|
||||||
*/
|
|
||||||
int ( * read ) ( struct spi_device *device, unsigned int address,
|
|
||||||
void *data, size_t len );
|
|
||||||
/** Write data to device
|
|
||||||
*
|
|
||||||
* @v device SPI device
|
|
||||||
* @v address Address to which to write
|
|
||||||
* @v data Data buffer
|
|
||||||
* @v len Length of data buffer
|
|
||||||
* @ret rc Return status code
|
|
||||||
*/
|
|
||||||
int ( * write ) ( struct spi_device *device, unsigned int address,
|
|
||||||
const void *data, size_t len );
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
static inline __attribute__ (( always_inline )) struct spi_device *
|
||||||
* @defgroup spidevs SPI device types
|
nvs_to_spi ( struct nvs_device *nvs ) {
|
||||||
* @{
|
return container_of ( nvs, struct spi_device, nvs );
|
||||||
*/
|
}
|
||||||
|
|
||||||
/** Atmel AT25010 serial EEPROM */
|
|
||||||
#define AT25010 { \
|
|
||||||
.word_len = 8, \
|
|
||||||
.size = 128, \
|
|
||||||
.block_size = 8, \
|
|
||||||
.command_len = 8, \
|
|
||||||
.address_len = 8, \
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @} */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An SPI device
|
|
||||||
*
|
|
||||||
* This data structure represents a real, physical SPI device attached
|
|
||||||
* to an SPI controller. It comprises the device type plus
|
|
||||||
* instantiation-specific information such as the slave number.
|
|
||||||
*/
|
|
||||||
struct spi_device {
|
|
||||||
/** SPI device type */
|
|
||||||
struct spi_device_type *type;
|
|
||||||
/** SPI bus to which device is attached */
|
|
||||||
struct spi_bus *bus;
|
|
||||||
/** Slave number */
|
|
||||||
unsigned int slave;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An SPI bus
|
* An SPI bus
|
||||||
*
|
*
|
||||||
*
|
* This data structure represents an SPI bus controller capable of
|
||||||
|
* issuing commands to attached SPI devices.
|
||||||
*/
|
*/
|
||||||
struct spi_bus {
|
struct spi_bus {
|
||||||
/** SPI interface mode
|
/** SPI interface mode
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <gpxe/spi.h>
|
#include <gpxe/spi.h>
|
||||||
|
#include <gpxe/bitbash.h>
|
||||||
|
|
||||||
/** A bit-bashing SPI bus */
|
/** A bit-bashing SPI bus */
|
||||||
struct spi_bit_basher {
|
struct spi_bit_basher {
|
||||||
|
|
|
@ -22,40 +22,48 @@
|
||||||
|
|
||||||
/** @} */
|
/** @} */
|
||||||
|
|
||||||
|
extern int threewire_read ( struct nvs_device *nvs, unsigned int address,
|
||||||
|
void *data, size_t len );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @defgroup spidevs SPI device types
|
* @defgroup tdevs Three-wire device types
|
||||||
* @{
|
* @{
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** Atmel AT93C46 serial EEPROM
|
static inline __attribute__ (( always_inline )) void
|
||||||
*
|
init_at93cx6 ( struct spi_device *device, unsigned int organisation ) {
|
||||||
* @v org Word size (8 or 16)
|
device->nvs.word_len = organisation;
|
||||||
*/
|
device->nvs.block_size = 1;
|
||||||
#define AT93C46( org ) { \
|
device->command_len = 3,
|
||||||
.word_len = (org), \
|
device->nvs.read = threewire_read;
|
||||||
.size = ( 1024 / (org) ), \
|
}
|
||||||
.block_size = 1, \
|
|
||||||
.command_len = 3, \
|
|
||||||
.address_len = ( ( (org) == 8 ) ? 7 : 6 ), \
|
|
||||||
.read = threewire_read, \
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Atmel AT93C56 serial EEPROM
|
/**
|
||||||
|
* Initialise Atmel AT93C46 serial EEPROM
|
||||||
*
|
*
|
||||||
* @v org Word size (8 or 16)
|
* @v device SPI device
|
||||||
|
* @v organisation Word organisation (8 or 16)
|
||||||
*/
|
*/
|
||||||
#define AT93C56( org ) { \
|
static inline __attribute__ (( always_inline )) void
|
||||||
.word_len = (org), \
|
init_at93c46 ( struct spi_device *device, unsigned int organisation ) {
|
||||||
.size = ( 2048 / (org) ), \
|
device->nvs.size = ( 1024 / organisation );
|
||||||
.block_size = 1, \
|
device->address_len = ( ( organisation == 8 ) ? 7 : 6 );
|
||||||
.command_len = 3, \
|
init_at93cx6 ( device, organisation );
|
||||||
.address_len = ( ( (org) == 8 ) ? 9 : 8 ), \
|
}
|
||||||
.read = threewire_read, \
|
|
||||||
}
|
/**
|
||||||
|
* Initialise Atmel AT93C56 serial EEPROM
|
||||||
|
*
|
||||||
|
* @v device SPI device
|
||||||
|
* @v organisation Word organisation (8 or 16)
|
||||||
|
*/
|
||||||
|
static inline __attribute__ (( always_inline )) void
|
||||||
|
init_at93c56 ( struct spi_device *device, unsigned int organisation ) {
|
||||||
|
device->nvs.size = ( 2048 / organisation );
|
||||||
|
device->address_len = ( ( organisation == 8 ) ? 9 : 8 );
|
||||||
|
init_at93cx6 ( device, organisation );
|
||||||
|
}
|
||||||
|
|
||||||
/** @} */
|
/** @} */
|
||||||
|
|
||||||
extern int threewire_read ( struct spi_device *device, unsigned int address,
|
|
||||||
void *data, size_t len );
|
|
||||||
|
|
||||||
#endif /* _GPXE_THREEWIRE_H */
|
#endif /* _GPXE_THREEWIRE_H */
|
||||||
|
|
Loading…
Reference in New Issue