ATtiny88初体验(七):TWI
TWI模块介绍
ATtiny88的TWI模块兼容Phillips I2C以及SMBus,支持主从模式,支持7bit地址,最大允许128个不同的从机地址。在多主机模式下,支持总线仲裁。从机模式下的数据速率高达400kHz,且从机地址可编程。在睡眠模式下,支持地址识别唤醒。
data:image/s3,"s3://crabby-images/1b92f/1b92f06d113cefaf124a74575c7a51501a8f87ba" alt="image.png"
注意:为了使用TWI模块, PRR
寄存器中的 PRTWI
位必须设为0。
数据位传输:
data:image/s3,"s3://crabby-images/f7bdf/f7bdfbfaa92b2759fee6e788f4e37708dff6934c" alt="image.png"
开始和停止条件:
data:image/s3,"s3://crabby-images/cc327/cc327d51cad5593d3bb43ca47f843ecc1d69938c" alt="image.png"
地址帧格式:
data:image/s3,"s3://crabby-images/6e47d/6e47d76488cb3dfcf5b948be6a2eafaa1f5efdc5" alt="image.png"
数据帧格式:
data:image/s3,"s3://crabby-images/12a2d/12a2dee8056e04b9bdc41ca1b6a7e431f0be24bb" alt="image.png"
完整的传输过程:
data:image/s3,"s3://crabby-images/05ff2/05ff2533e885ba3e45402153a7eb71a2bfbfa8c7" alt="image.png"
普通模式时钟频率:
\[f_{SCL} = \frac{clk_{I/O}}{16 + (2 \times TWBR \times TWPS)}\]
高速模式时钟频率:
\[f_{SCL} = \frac{clk_{TWIHS}}{16 + (2 \times TWBR \times TWPS)}\]
其中, \(clk_{I/O}\) 为分频后的系统时钟, \(clk_{TWIHS}\) 为系统时钟。
注意:主机模式下, TWBR
值不能低于10。
下图展现了一个典型传输过程中,应用程序如何与TWI模块交互的。
data:image/s3,"s3://crabby-images/72010/720108019a7e3e67cf96aa440010d64d7de7a997" alt="image.png"
ATtiny88的TWI拥有四种模式:Master Transmitter、Master Receiver、Slave Transmitter、Slave Receiver。
Master Transmitter模式下的状态码:
data:image/s3,"s3://crabby-images/f7347/f7347e77926c95a07f44ae35198d7380aabebc6c" alt="image.png"
data:image/s3,"s3://crabby-images/37c76/37c7614f75419b8b8be2621e844f80db9b6482e5" alt="image.png"
Master Receiver模式下的状态码:
data:image/s3,"s3://crabby-images/ac5e7/ac5e7f795bc13c1004627fbceedf9ccd40f57db2" alt="image.png"
data:image/s3,"s3://crabby-images/50cff/50cff0032f632cf0f6f31b93ff8b8dea773a318d" alt="image.png"
Slave Receiver模式下的状态码:
data:image/s3,"s3://crabby-images/aa01e/aa01e991951e1b5bb434d7c2201f25982438d191" alt="image.png"
data:image/s3,"s3://crabby-images/0f477/0f477e909f1cc1d16c0d1e853d3ebb89dd95be00" alt="image.png"
Slave Transmitter模式下的状态码:
data:image/s3,"s3://crabby-images/303b8/303b83a80a1a6be4dadf4361504af44e4316071a" alt="image.png"
data:image/s3,"s3://crabby-images/1d155/1d1559f2c49570c99b59de2eecd1d31a0ad75168" alt="image.png"
其他状态码:
data:image/s3,"s3://crabby-images/d5c44/d5c44d46af4a637560609d165260402b53b06a72" alt="image.png"
多主机仲裁过程及状态码:
data:image/s3,"s3://crabby-images/3bf80/3bf803dbd03252eba2faac37db0c7d2ff478a02c" alt="image.png"
寄存器
data:image/s3,"s3://crabby-images/d4116/d41167738de123ae37426ace9f5dd2ce727e35c6" alt="image.png"
TWBR[7:0]
:分频系数,在主机模式下不能低于10。
data:image/s3,"s3://crabby-images/15373/15373a9c52575b4073875b911e73ae28d40aaf0d" alt="image.png"
TWINT
:TWI中断标志,必须写1清除(不会执行完中断自动清除),在清除前,必须完成对 TWAR
寄存器、 TWSR
寄存器、 TWDR
寄存器的操作。 TWINT
置位期间,SCL线始终保持低电平。
TWEA
:写1使能应答。
TWSTA
:写1产生START信号,在START信号传输完成后必须手动清除该位。
TWSTO
:写1产生STOP信号,该位会自动清除。从机状态下,设置该位可以从错误状态中恢复。
TWWC
:写冲突标志,在 TWINT
位为1时写 TWDR
寄存器清除。
TWEN
:写1使能TWI模块。
TWIE
:写1使能TWI中断。
data:image/s3,"s3://crabby-images/f692e/f692ec0e0dc98933f013b54209646b71910a9c69" alt="image.png"
TWS
:TWI状态码。
TWPS
:TWI分频。
data:image/s3,"s3://crabby-images/a8680/a868010b15293096ddc4f2dd6267ff831cbcd10f" alt="image.png"
data:image/s3,"s3://crabby-images/27eb4/27eb48245d999201cab05ae24211e8169d6910d6" alt="image.png"
data:image/s3,"s3://crabby-images/42735/42735cc438d215161e6b5f41fbe0d6f6bc57b7a5" alt="image.png"
TWA[6:0]
:TWI从机地址。
TWGCE
:写1使能General Call。
data:image/s3,"s3://crabby-images/1e017/1e017d427f0c18c219404d77b907e7a4df2720fe" alt="image.png"
TWAM[6:0]
:TWI地址掩码,设为1将忽略对应位的匹配。
data:image/s3,"s3://crabby-images/72c5c/72c5c198ced7732ef52ab43bc7ae677a21eee259" alt="image.png"
data:image/s3,"s3://crabby-images/b7851/b785158831f15d49bf02e7a977bc2dfb3cf989a1" alt="image.png"
代码
下面的代码展示了如何使用ATtiny88的TWI模块与SSD1306 OLED进行通信刷屏。
源文件的组织结构如下:
| .
├── Makefile
├── inc
│ ├── serial.h
│ └── serial_stdio.h
└── src
├── main.c
├── serial.c
└── serial_stdio.c
|
src/main.c
源文件的内容如下:
src/main.c |
---|
| #include <stdint.h>
#include <stdio.h>
#include <avr/io.h>
#include <avr/interrupt.h>
#include <serial_stdio.h>
#define OLED_ADDR 0x3C
#define oled_write_command(cmd) oled_write_byte(0, cmd)
#define oled_write_data(data) oled_write_byte(1, data)
static void oled_setup(void);
static void oled_write_byte(uint8_t dc, uint8_t data);
static void oled_fill(uint8_t color);
int main(void)
{
cli();
stdio_setup(); // initialize stdio and redirect it to serial
oled_setup(); // initialize oled
sei();
static const uint8_t colors[] = {
0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40,
0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02
};
uint8_t i = 0;
for (;;) {
oled_fill(colors[i]);
i = (i + 1) % sizeof(colors);
}
}
static void oled_setup(void)
{
static const uint8_t cmds[] = {
0xAE, 0xD5, 0x80, 0xA8, 0x3F,
0xD3, 0x00, 0x40, 0x8D, 0x14,
0x20, 0x00, 0xA1, 0xC8, 0xDA,
0x12, 0x81, 0xEF, 0xD9, 0xF1,
0xDB, 0x30, 0xA4, 0xA6, 0xAF
};
TWSR = 0x00; // TWPS = 1
TWBR = 12; // TWBR = 12, freq = 16MHz / (16 + 2 * 12 * 1) = 400KHz
TWHSR = 0x00; // disable high speed mode
for (uint8_t i = 0; i < sizeof(cmds); i++) {
oled_write_command(cmds[i]);
}
oled_fill(0x00);
}
static void oled_write_byte(uint8_t dc, uint8_t data)
{
// transmit START condition
TWCR = _BV(TWINT) | _BV(TWSTA) | _BV(TWEN);
while (!(TWCR & _BV(TWINT)));
// transmit SLA+W
TWDR = OLED_ADDR << 1;
TWCR = _BV(TWINT) | _BV(TWEN);
while (!(TWCR & _BV(TWINT)));
// transmit control byte
TWDR = dc ? 0x40 : 0x00;
TWCR = _BV(TWINT) | _BV(TWEN);
while (!(TWCR & _BV(TWINT)));
// transmit data byte
TWDR = data;
TWCR = _BV(TWINT) | _BV(TWEN);
while (!(TWCR & _BV(TWINT)));
// transmit STOP condition
TWCR = _BV(TWINT) | _BV(TWSTO) | _BV(TWEN);
// Note that TWINT is NOT set after a STOP condition has been transmitted
}
static void oled_fill(uint8_t color)
{
oled_write_command(0x21);
oled_write_command(0x00);
oled_write_command(0x7F);
oled_write_command(0x22);
oled_write_command(0x00);
oled_write_command(0x07);
for (uint8_t i = 0; i < 128; i++) {
for (uint8_t j = 0; j < 8; j++) {
oled_write_data(color);
}
}
}
|
参考资料
- ATtiny88 Datasheet