首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >ATmega328 + SPI闪存

ATmega328 + SPI闪存
EN

Stack Overflow用户
提问于 2010-10-20 18:09:11
回答 1查看 12K关注 0票数 6

前言:这个问题是关于我和我的大学教授一起做的一个项目。这不是为了一个等级,但我在这位教授的名声确实很重要。因此,虽然我在这个项目上的成功对我来说很重要,但我并不认为寻求堆栈溢出的帮助是不公平的。

尽管如此,这里是我的项目的高级概述。我有一个ATmega328微控制器。我有一个微芯片SST 64 Mbit闪存卡。ATmega有一个SPI的硬件实现。闪存有SPI的硬件实现。

我的目标是在SPI主模式下使用ATmega从闪存芯片读取数据并将数据写入闪存芯片。内存是组织在一个多覆盖结构,这是一个很好的擦除,但就我的目的,它基本上只有32,768页页,每个256字节。

写数据的基本思想是我发送一个指令字节,然后是起始地址,然后是数据。要读取数据,基本思想是发送指令字节,然后发送起始地址,然后发送虚拟字节,然后开始发送数据。

这里是数据表:

微控制器:http://www.atmel.com/dyn/resources/prod_documents/doc8271.pdf

闪光灯:http://www.sst.com/dotAsset/40498.pdf

代码:

代码语言:javascript
复制
#include <SPI.h>
#include <Peggy2.h>

#define SS_PIN  16

Peggy2 frame1;
byte toDisp = 0;
byte checker = 0;

void setup()
{
    frame1.HardwareInit();
    pinMode(SS_PIN,OUTPUT); //set pin16 to output, SS pin
    SPI.setClockDivider(SPI_CLOCK_DIV2); //set the SPI clock to f/2, fastest possible
    SPI.begin();    //SPI lib function which sets ddr for SCK and MOSI pin
                    //MISO is auto input
                    //see SPI.cpp for more info

}

void loop()
{

    if(!checker){
                enableProgramming();
        programData();
        toDisp = receiveByte(0);
        checker = 1;
        frame1.WriteRow(0,toDisp);
    }
    frame1.RefreshAll(2);

}

byte receiveByte(unsigned long startAddress)
{
    //Begin High Speed Read Instruction
    //See p. 10 of SST data sheet
    digitalWrite(SS_PIN,LOW);
    SPI.transfer(0x0B); //high speed read instruction
    SPI.transfer(0x00); //next 3 transfers are address bits A32 - A0
    SPI.transfer(0x00); //So this will read the first byte on the chip
    SPI.transfer(0x00); //last address bits
    SPI.transfer(0xFF); //dummy byte is required to start sending data back to uP
    SPI.transfer(0xFF); //I'm hoping that if I transfer a bullshit byte, the flash
                        //chip will transfer it's data to me in the same time
    digitalWrite(SS_PIN,HIGH);
    //End High Speed Read Instruction   
    return SPDR;    
}

//will perform the read instruction starting from
//startAddress and will receive numOfBytes bytes in
//succession
void receiveBytes(int numOfBytes, unsigned long startAddress)
{
    digitalWrite(SS_PIN,LOW);
    SPI.transfer(0x0B);//high speed read instruction

}

//will perform:
// 1) Chip Erase
// and loop through:
// 1) Page Program
// 2) increment Page
//until the data has finished **note this can loop and over write beginning of memory
void programData(){
    //Begin ChipErase Instruction
    //See p. 17 of SST data sheet
    digitalWrite(SS_PIN,LOW);
    SPI.transfer(0x60);//chip erase instruction
    digitalWrite(SS_PIN,HIGH);
    delay(50);//spec'd time for CE to finish
                //don't bother polling because time to program is irrelevant
    //End ChipErase Instruction

        //Begin WREN Instruction
    //See p. 18 of SST data sheet
    digitalWrite(SS_PIN,LOW);
    SPI.transfer(0x06);//write enable instruction
    digitalWrite(SS_PIN,HIGH);
    //End WREN Instruction

    digitalWrite(SS_PIN,LOW);
    SPI.transfer(0x02); //page program instruction
    SPI.transfer(0x00); //first 8 address bits
    SPI.transfer(0x00); //2nd 8 address bits
    SPI.transfer(0x00); //3rd 8 address bits
    SPI.transfer(0xAA); //10101010 is the byte I should be writing
    digitalWrite(SS_PIN,HIGH);
    delayMicroseconds(3000); //wait 3 ms for page program


    /*
    //Begin Page-Program Instruction
    //see p. 13 of SST data sheet
    byte firstAddress = 0;
    byte secondAddress = 0;
    //this loop will write to every byte in the chips memory
    //32,768 pages of 256 bytes = 8,388,608 bytes
    for(unsigned int i = 0; i < 32,768; ++i) //long variable is number of pages
    {
        digitalWrite(SS_PIN,LOW);
        ++secondAddress; //cycles from 0 to 255, counts pages
        firstAddress = i>>8; // floor(i/256)

        SPI.transfer(0x02);//Page-Program instruction byte
        SPI.transfer(firstAddress); //increments every 256 pages i.e. at page 256 this should be 1
        SPI.transfer(secondAddress); //increments every 256 bytes, i.e every page
        SPI.transfer(0x00); //beginning of a page boundary
        for(int j = 0; j < 256; ++j) //number of bytes per page
        {
            SPI.transfer(2program[(256*i) + j]);//data byte transfer            
        }
        digitalWrite(SS_PIN,HIGH);
        delayMicroseconds(2500); //2500us (2.5ms) delay for each page-program instruction to execute
    }
    //End Page-Program Instruction
    */
}

//Will prepare the chip for writing by performing:
// 1) arm the status register
// 2) Write Enable instruction
//Only needs to be performed once!
void enableProgramming(){
    //Begin EWSR & WRSR Instructions
    //See p. 20 of SST data sheet for more info
    digitalWrite(SS_PIN,LOW); //lower the SS pin
    SPI.transfer(0x50); //enable write status register instruction
    digitalWrite(SS_PIN,HIGH); //raise the SS pin
    delay(10);
    digitalWrite(SS_PIN,LOW); //lower the SS pin
    SPI.transfer(0x01); //write the status register instruction
    SPI.transfer(0x00);//value to write to register
                //xx0000xx will remove all block protection
    digitalWrite(SS_PIN,HIGH);
    //End EWSR & WRSR Instructions

    //Begin WREN Instruction
    //See p. 18 of SST data sheet
    digitalWrite(SS_PIN,LOW);
    SPI.transfer(0x06);//write enable instruction
    digitalWrite(SS_PIN,HIGH);
    //End WREN Instruction

}

所以这应该是一个测试程序,它在闪存上编程1字节,然后读取它,并在我的LED阵列上显示这个字节。如果您对LED阵列感兴趣,可以在这里找到:http://evilmadscience.com/tinykitlist/157

我相信我的阅读功能有效,因为我第一次运行这个功能时,所有的8个LED都亮起来了。这会告诉我,当它处于所有1s的工厂状态时,它会读取闪存。很明显,我把写作搞砸了,因为点亮的字节和我想要编程的字节完全不对应。

我还应该指出,我正在为Arduino使用默认的SPI库,并且框架缓冲区函数可以工作。当我做frame1.WriteRow(toDisp),这是正确的工作,并已广泛测试。

如果有人有时间或耐心去弄清楚我做错了什么,那就太棒了。

编辑:以帮助调试:

LED由驱动芯片驱动,驱动芯片也使用SPI接口。这部分代码不是我写的。在示波器上,我可以看到SCK线是由代码的那一部分驱动的。然而,我也有一个关于MOSI引脚的探针,如果我不点亮任何灯,它似乎永远不会高。对我来说,这意味着我发送的信息不正确。也就是..。也许我的SPI.transfer()需要一个启用函数什么的?

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2011-06-13 19:46:35

对于任何仍然好奇的人来说,问题在于内存芯片对缓慢上升的时间非常敏感。在放入施密特扳机后,一切都完美无缺。

票数 5
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/3980939

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档