- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
前言:这个问题是关于我正在与我的大学教授合作的一个项目。这不是为了成绩,但我在这位教授中的声誉确实很重要。因此,虽然我在这个项目上的成功对我来说很重要,但我并不认为向 Stack Overflow 寻求帮助是不公平的。
也就是说,这是一个高级 我的项目概述 .我有一个 ATmega328 微 Controller 。我有一个 Microchip SST 64 Mbit 闪存卡。 ATmega 具有 SPI 的硬件实现。闪存具有 SPI 的硬件实现。
我的目标是在 SPI 主模式下使用 ATmega 从闪存芯片读取数据并将数据写入闪存芯片。内存被组织在一个多重覆盖结构中,这很适合删除,但就我的目的而言,它基本上只有 32,768 页,每页 256 字节。
写数据的基本思想是我发送一个指令字节,然后是起始地址,然后是数据。读取数据的基本思想是我发送一个指令字节,然后是起始地址,然后是一个虚拟字节,然后它开始向我发送数据。
以下是数据表:
微 Controller :http://www.atmel.com/dyn/resources/prod_documents/doc8271.pdf
闪:http://www.sst.com/dotAsset/40498.pdf
代码:
#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
}
frame1.WriteRow(toDisp)
,它工作正常并且已经过广泛测试。
SPI.transfer()
需要启用功能还是什么?
最佳答案
对于任何仍然好奇的人来说,问题是内存芯片对缓慢的上升时间极其敏感。放入施密特触发器后,一切正常。
关于arduino - ATmega328 + SPI 闪存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3980939/
我的代码用于 atmega32 微 Controller ,用于将坐标转换为 nmea 格式,我如何将其转换为 atmega16 代码。因为它使用的是atmega32库文件该代码可以在 atmega1
在某些情况下,我正在为 Arduino 项目开发基本上是“Simon”的东西。我正在使用 ATMEGA2560 并使用 Atmel Studio 6.1 编辑代码。该项目使用随机数生成器来创建按钮输入
我有一个问题,我必须用 USART 按下我电脑上的键来控制 ATMEGA 1280 的 PWM。我可以使用 ASCII 键和控制 PWM 来控制它。问题是他们要求使用箭头键,现在问题是箭头键没有 AS
我正在尝试生成 16kHz pwm...这是我现在正在使用的代码。 ` int main(void){ DDRD |= (1 << DDD6); // PD6 is now an output OCR
我正在尝试进行数据传输控制。一个带有上拉电阻的按钮连接到 ATmega644P(Slave) 的 PINA3。如果只有总线空闲,我想在按下按钮时发送一些数据。出于测试目的,我使用另一个 ATmega6
我开始在 atmega8 上的应用程序中实现类似于任务切换的东西。主要思想是有一个指向“当前屏幕”结构的指针。 “当前屏幕”包含用于刷新屏幕、处理按钮和中断的处理程序。 不幸的是,我发现更改函数指针是
在下面的代码图中,按位左移运算符与寄存器一起使用来进行串口初始化,当寄存器位(RXEN, TXEN)未初始化 enter image description here 这里是寄存器描述和代码的链接 h
我有 3 个完全相同的 PCB 容纳 ATmega2560 MCU。我将 Atmel Studio 7 用于 fuse 和闪烁 HEX。我正在使用 Visual Studio 和 Visual Mic
一段时间以来,我一直在研究如何使用我的 atmega32 以快速 pwm 模式控制电机(控制其速度)。我需要使用 8 位 Timer0,因为我对其他计数器有其他用途。我想我知道如何为此任务初始化计时器
我想将atmega32与MMC/SD卡连接,但我有问题是: #define F_CPU 8000000UL void uart_init(unsigned int BAUD) unsigned
我正在尝试在不使用延迟功能的情况下使 LED 闪烁。我遇到了使用定时器中断并尝试了它,它编译得很好。但输出固定为 PORTA = 0x01;所以,我认为 ISR 功能不起作用。代码中我缺少什么吗?谢谢
我的问题是关于实时数据记录和多中断。我正在尝试通过 winAVR 对 MCU-ATMega 1280 进行编程,使其从正交编码器(20um/pitch)读取脉冲并将数据存储到闪存(Microchip
我想要完成的事情:用户在 iOS 应用程序上输入密码。如果密码正确,则向微处理器发送一位值,表明用户获得了正确的密码。然后微处理器可以启动步进电机打开门闩。 有没有人尝试过实现这个目标?如果是这样,你
我有一个 atmega 1284p 的问题,我写了这个 ISR 来通过串口接收命令。如果我将 CR 或 LF 发送到命令的末尾,程序将正常运行,但如果我不发送它们中的任何一个,程序将停留在 ISR 中
我有几个关于 AVR (atmega 8) 中 SPI 的问题。我想传输字节 0xfe ,但功能看起来像这样: void sendDataSpi(char data ) 这是否意味着我必须将我的整数(
我是 Atmegas ant AVR 编程的新手,所以我希望你能原谅我这个菜鸟问题:我有这段代码,它可以在 1MHZ 下完美运行,但是如果我在更改 F_CPU 频率的同时将 Atmega 融合到内部
我正在使用 avr-libc 开发 C 应用程序在 AVR 上 ATmega328P微 Controller 。因为我没有它的 ICE 调试器,所以我遵循了 these instructions和 t
#include #include #define BAUDRATE 115200 #define BAUD_PRESCALLER (((F_CPU / (BAUDRATE * 16UL))) -
我不知道为什么,LED 脉冲并没有增加亮度,而是每个脉冲之间的周期越来越短。这是从教程中复制的代码,在他的视频中效果很好,但对我来说效果不佳,即使在模拟器中也是如此。怎么会发生这种事? 使用 avr
我已经编写了一些代码来向 usart 输出一条 9 位消息。 代码如下: bool_t SERIAL__TX_SEND_NINE(U8_t ch, bool_t nine) // se
我是一名优秀的程序员,十分优秀!