Utilizzare una EEPROM 24WC16P con Arduino

Ciao a tutti, vorrei iniziare ad utilizzare una EEPROM 24WC16P (recuperata da vecchi progetti) con Arduino, utilizzando la libreria Wire.h.

Ho cercato un pò in rete (e anche nel forum) ma non ho trovato molto (soprattutto in italiano). In particolare si trovano solo esempi riguardanti EEPROM più grandi (256).

C'è qualcuno che ci ha già provato?

funziona allo stesso modo solo ricorda che hai meno indirizzi... lo sketch scritto sul playground funziona ma ti ripeto che devi ridurre gli indirizzi calcoli quanta memoria ha, e converti in byte ottenedo il numero di indirizzi che contiene

ps ricorda che il primo indirizzo è 0 non 1 quindi non fiirà ad esempio in 16(memoria da 16byte) ma con 15 ;)

ciao

Ciao 83darking83

Il tuo EEprom é di 16k bit e percui 2048 Byte

http://pdf1.alldatasheet.com/datasheet-pdf/view/57845/CATALYST/CAT24WC16P-1.8TE13D.html

Ciao Uwe

Ok, intanto ringrazio entrambi per le celeri risposte! :slight_smile:

Ho connesso la EEPROM secondo il seguente schema:

Pin 1 → GND (Arduino)
Pin 2 → GND (Arduino)
Pin 3 → GND (Arduino)
Pin 4 → GND (Arduino)
Pin 8 → 5 V (Arduino)
Pin 7 → GND (Arduino)
Pin 6 → Analog Pin 5 (Arduino)
Pin 5 → Analog Pin 4 (Arduino)

Tuttavia, inserendo il seguente codice:

#include <Wire.h>
#include <LiquidCrystal.h>


#define EEPROM_ADDRESS 0x50
LiquidCrystal lcd(7, 6, 5, 4, 3, 2);


void i2c_eeprom_write_byte( int deviceaddress, unsigned int eeaddress, byte data ) 
{
  int rdata = data;
  Wire.beginTransmission(deviceaddress);
  Wire.send((int)(eeaddress >> 8)); // MSB
  Wire.send((int)(eeaddress & 0xFF)); // LSB
  Wire.send(rdata);
  Wire.endTransmission();
}

// WARNING: address is a page address, 6-bit end will wrap around
// also, data can be maximum of about 30 bytes, because the Wire library has a buffer of 32 bytes
void i2c_eeprom_write_page( int deviceaddress, unsigned int eeaddresspage, byte* data, byte length ) 
{
  Wire.beginTransmission(deviceaddress);
  Wire.send((int)(eeaddresspage >> 8)); // MSB
  Wire.send((int)(eeaddresspage & 0xFF)); // LSB
  byte c;
  for ( c = 0; c < length; c++)
    Wire.send(data[c]);
  Wire.endTransmission();
}

byte i2c_eeprom_read_byte( int deviceaddress, unsigned int eeaddress ) 
{
  byte rdata = 0xFF;
  Wire.beginTransmission(deviceaddress);
  Wire.send((int)(eeaddress >> 8)); // MSB
  Wire.send((int)(eeaddress & 0xFF)); // LSB
  Wire.endTransmission();
  Wire.requestFrom(deviceaddress,1);
  if (Wire.available()) rdata = Wire.receive();
  return rdata;
}

// maybe let's not read more than 30 or 32 bytes at a time!
void i2c_eeprom_read_buffer( int deviceaddress, unsigned int eeaddress, byte *buffer, int length ) {
  Wire.beginTransmission(deviceaddress);
  Wire.send((int)(eeaddress >> 8)); // MSB
  Wire.send((int)(eeaddress & 0xFF)); // LSB
  Wire.endTransmission();
  Wire.requestFrom(deviceaddress,length);
  int c = 0;
  for ( c = 0; c < length; c++ )
    if (Wire.available()) buffer[c] = Wire.receive();
}

int temp = 0;

void setup(void)
{
  lcd.begin(20, 4);
  lcd.clear();
  lcd.print("EEPROM Test");
  Wire.begin();
  i2c_eeprom_write_byte( EEPROM_ADDRESS, 0, 200 );
  delay(20);
}

void loop(void)
{
  temp = i2c_eeprom_read_byte( EEPROM_ADDRESS, 0 );
  lcd.setCursor(0,1);
  lcd.print(temp);
  delay(8000);
}

sul display appare sempre:

EEPROM TEST
255

Cosa sbaglio?

Se mi chiedi cosi... :-?

leggi bene http://arduino.cc/en/Reference/Wire dove é scritto: This library allows you to communicate with I2C / TWI devices. On most Arduino boards, SDA (data line) is on analog input pin 4, and SCL (clock line) is on analog input pin 5.

Ciao Uwe

Avevo dato per scontato che 4 e 5 erano analog pin... :o

Comunque modifico il mio post precedente, onde evitare confusioni...

Quindi i pin 4 e 5 analogici funzionano anche in uscita con una comunicazione I2C

ciao 83darking83
"Avevo dato per scontato che 4 e 5 erano analog pin… " allora é giusto.
Servono resistenze di pullup sui SDA e SCL. La libreria WIRE.H mette quelle interne e per cavetti corti non serve aggiungere altre sesistenze
(Wire library
Wire library will enable ATMEL internal Pullup’s. So using additional pullup’s with a short bus may not be required, for example interface only a single nearby EEPROM. )

c’é anche la libreria http://www.arduino.cc/en/Reference/EEPROM

o http://www.arduino.cc/playground/Main/InterfacingWithHardware#Storage

Ricontrollando mi sembra che hai collegato la EEPROM sbagliato.

EEprom
Pin 1 (A0) → GND (Arduino)
Pin 2 (A1) → GND (Arduino)
Pin 3 (A2) → GND (Arduino)
Pin 4 (GND) → GND (Arduino)
Pin 5 (SDA) → Analog Pin 4 (Arduino)
Pin 6 (SCL) → Analog Pin 5 (Arduino)
Pin 7 (WP) → GND (Arduino)
Pin 8 (Vpp) → 5 V (Arduino)

http://pdf1.alldatasheet.com/datasheet-pdf/view/57845/CATALYST/CAT24WC16P-1.8TE13D.html

Ciao Uwe

Hai ragione, ho sbagliato a scrivere la lista dei pin, in realtà ho collegato l'EEPROM correttamente. Non mi resta che provare con le resistenze di Pull-up....

Niente, non va neanche con le resistenze di pull up (da 10KOHM) e neanche sostituendo la EEPROM!

Non so più che fare...

allora per aiutarti ci provo anch'io ho trovato in mezzo alla roba una 20WC16 però non ho capito una cosa: l tuo EEprom é di 16k bit e percui 2048 Byte cosa centrano i 2024 byte

2024?

scusa 2048 byte

16 Kbit / 8 = 2 KByte * 1024 = 2048 byte ;)

Ogni aiuto è ben accetto! :)

io ti posto il codice che ho usato(ai tempi) per capirne il funzionamento

non so a chi rendere grazie in quanto non mi ricordo da che pagina l’ho preso, spero che l’autore non si arrabbi:

#include <Wire.h>

/* Simple read & write to a 24LC256 256K EEPROM using the Wire library.  
Addresses are ints - 0000-7FFF (32767) Data is bytes (8 bits x 32767 = 256K) 
Functions for R/W of single byte or a page of bytes. Max page is 28 bytes. 
Arduino GND- A0-|oU |-Vcc to Arduino Vcc 
Arduino GND- A1-| |-WP to GND for now. Set to Vcc for write protection. 
Arduino GND- A2-| |-SCL to Arduino 5 
Arduino GND-Vss-| |-SDA to Arduino 4 
--- (A2, A1, A0 to GND for 1010000 (0x50) address.) 
If set to Vcc adds to address (1010,A2,A1,A0) */


#define EEPROM_ADDR 0x50 // I2C Buss address of 24LC256 256K EEPROM 
void setup(){ 
  Wire.begin(); // join I2C bus (address optional for master) 
Serial.begin(9600);
Serial.println("\n----------------\n"); // TESTS FOR EACH FUNCTION BEGIN HERE 
Serial.println("Writing Test:");

for (int i=0; i<32; i++){ // loop for first 20 slots
  i2c_eeprom_write_byte(EEPROM_ADDR,i,77-i);// write address + 65 A or 97 a 
Serial.print(". ");
delay(5); // NEED THIS DELAY! (tests suggest it can be as // small as delay(3) -- BBR 
} 
Serial.println("");
delay(500); 
Serial.println("Reading Test:"); 
for (int i=0; i<32; i++){ // loop for first 20 slots 
Serial.print(i2c_eeprom_read_byte(EEPROM_ADDR, i),BYTE);
Serial.print(" "); } // setup for page tests . . . 
byte PageData[30]; // array that will hold test data for a page 
byte PageRead[30];
// array that will hold result of data for a page 
for (int i=0; i<30; i++){ // zero both arrays for next test 
PageData[i] = 0;
PageRead[i] = 0;
} 
Serial.println(""); 
for (int i=0; i<30; i++) PageData[i] = i+33; // fill up array for next test char 33 = ! 
Serial.println("Writing Page Test:");
Serial.println(millis());
i2c_eeprom_write_page(EEPROM_ADDR, 100, PageData, 28 ); // 28 bytes/page is max 
Serial.println(millis());
Serial.println("Reading Page Test:");
i2c_eeprom_read_buffer( EEPROM_ADDR, 100, PageRead, 28);
for (int i=0; i<28; i++){
  Serial.print(PageRead[i],BYTE); // display the array read 
Serial.print(" "); 
}
}

void loop(){} 
  
  void i2c_eeprom_write_byte( int deviceaddress, unsigned int eeaddress, byte data ) 
{
  int rdata = data; 
Wire.beginTransmission(deviceaddress); 
Wire.send((int)(eeaddress >> 8)); // Address High Byte 
Wire.send((int)(eeaddress & 0xFF)); // Address Low Byte 
Wire.send(rdata); 
Wire.endTransmission();
}
// Address is a page address, 6-bit (63). More and end will wrap around 
// But data can be maximum of 28 bytes, because the Wire library has a buffer of 32 bytes 
void i2c_eeprom_write_page( int deviceaddress, unsigned int eeaddresspage, byte* data, byte length )
{
  Wire.beginTransmission(deviceaddress);
  Wire.send((int)(eeaddresspage >> 8)); // Address High Byte 
Wire.send((int)(eeaddresspage & 0xFF)); // Address Low Byte 
byte c; 
for ( c = 0; c < length; c++) Wire.send(data[c]); 
Wire.endTransmission(); 
delay(10); // need some delay 
}
byte i2c_eeprom_read_byte( int deviceaddress, unsigned int eeaddress ) 
{
  byte rdata = 0xFF; 
Wire.beginTransmission(deviceaddress); 
Wire.send((int)(eeaddress >> 8)); // Address High Byte 
Wire.send((int)(eeaddress & 0xFF)); // Address Low Byte 
Wire.endTransmission(); 
Wire.requestFrom(deviceaddress,1); 
if (Wire.available()) rdata = Wire.receive();
return rdata; } // should not read more than 28 bytes at a time! 

void i2c_eeprom_read_buffer( int deviceaddress, unsigned int eeaddress, byte *buffer, int length ) 
{
  Wire.beginTransmission(deviceaddress);
  Wire.send((int)(eeaddress >> 8)); // Address High Byte
  Wire.send((int)(eeaddress & 0xFF)); // Address Low Byte
  Wire.endTransmission();
  Wire.requestFrom(deviceaddress,length); //
  int c = 0;
  for ( int c = 0; c < length; c++ ) 
if (Wire.available()) buffer[c] = Wire.receive();
}

@garinus il tuo codice rispedisce indietro 255(ÿ), probabilmente a questo punto peso sia un problema di collegamento o lo sketch ha gli indirizzi sbagliati. Ho provato con una24WC16P e una 24c16 che sono uguali come memoria e da sempre 255

Esatto! Hai il mio stesso problema!
Ritorna sempre 255, maledizione!

Secondo me, guardando il codice, dipende dal fatto che non c’è comunicazione con la EEPROM:

byte rdata = 0xFF;
  if (Wire.available()) rdata = Wire.receive();
return rdata;

Infatti, (0xFF)16 = (255)10

io ho provato per vedere se ci fosse comunicazione cosi:

if (Wire.available()) 
{
rdata = Wire.receive();
Serial.print("aviable");
}
return rdata;

e cosi non mi entrava mai nell'if

allora avrete il codice della memoria diverso.. bisogna cercare nel datasheet

con quel codice ho provato anche su una atmel 24c16 e non funziona forse avro sbagliato piedinatura ci controllo