machine/esp32c3: implement i2c for esp32-c3
Этот коммит содержится в:
родитель
803ba4f54d
коммит
94459cefe5
2 изменённых файлов: 326 добавлений и 1 удалений
|
@ -1,4 +1,4 @@
|
|||
//go:build atmega || nrf || sam || stm32 || fe310 || k210 || rp2040 || mimxrt1062
|
||||
//go:build atmega || nrf || sam || stm32 || fe310 || k210 || rp2040 || mimxrt1062 || esp32c3
|
||||
|
||||
package machine
|
||||
|
||||
|
|
|
@ -111,6 +111,10 @@ func (p Pin) outFunc() *volatile.Register32 {
|
|||
return (*volatile.Register32)(unsafe.Add(unsafe.Pointer(&esp.GPIO.FUNC0_OUT_SEL_CFG), uintptr(p)*4))
|
||||
}
|
||||
|
||||
func (p Pin) pinReg() *volatile.Register32 {
|
||||
return (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(&esp.GPIO.PIN0)) + uintptr(p)*4)))
|
||||
}
|
||||
|
||||
// inFunc returns the FUNCy_IN_SEL_CFG register used for configuring the input
|
||||
// function selection.
|
||||
func inFunc(signal uint32) *volatile.Register32 {
|
||||
|
@ -596,3 +600,324 @@ func (usbdev *USB_DEVICE) flush() {
|
|||
for usbdev.Bus.GetEP1_CONF_SERIAL_IN_EP_DATA_FREE() == 0 {
|
||||
}
|
||||
}
|
||||
|
||||
// I2C code
|
||||
|
||||
var (
|
||||
I2C0 = &I2C{}
|
||||
)
|
||||
|
||||
type I2C struct{}
|
||||
|
||||
// I2CConfig is used to store config info for I2C.
|
||||
type I2CConfig struct {
|
||||
Frequency uint32 // in Hz
|
||||
SCL Pin
|
||||
SDA Pin
|
||||
}
|
||||
|
||||
const (
|
||||
clkXTAL = 0
|
||||
clkFOSC = 1
|
||||
clkXTALFrequency = uint32(40e6)
|
||||
clkFOSCFrequency = uint32(17.5e6)
|
||||
i2cClkSourceFrequency = clkXTALFrequency
|
||||
i2cClkSource = clkXTAL
|
||||
)
|
||||
|
||||
func (i2c *I2C) Configure(config I2CConfig) error {
|
||||
i2c.initClock(config)
|
||||
i2c.initNoiseFilter()
|
||||
i2c.initPins(config)
|
||||
i2c.initFrequency(config)
|
||||
i2c.startMaster()
|
||||
return nil
|
||||
}
|
||||
|
||||
// go: inline
|
||||
func (i2c *I2C) initClock(config I2CConfig) {
|
||||
// reset I2C clock
|
||||
esp.SYSTEM.SetPERIP_RST_EN0_EXT0_RST(1)
|
||||
esp.SYSTEM.SetPERIP_CLK_EN0_EXT0_CLK_EN(1)
|
||||
esp.SYSTEM.SetPERIP_RST_EN0_EXT0_RST(0)
|
||||
// disable interrupts
|
||||
esp.I2C.INT_ENA.ClearBits(0x3fff)
|
||||
esp.I2C.INT_CLR.ClearBits(0x3fff)
|
||||
|
||||
esp.I2C.SetCLK_CONF_SCLK_SEL(i2cClkSource)
|
||||
esp.I2C.SetCLK_CONF_SCLK_ACTIVE(1)
|
||||
esp.I2C.SetCLK_CONF_SCLK_DIV_NUM(i2cClkSourceFrequency / (config.Frequency * 1024))
|
||||
esp.I2C.SetCTR_CLK_EN(1)
|
||||
}
|
||||
|
||||
// go: inline
|
||||
func (i2c *I2C) initNoiseFilter() {
|
||||
esp.I2C.FILTER_CFG.Set(0x377)
|
||||
}
|
||||
|
||||
// go: inline
|
||||
func (i2c *I2C) initPins(config I2CConfig) {
|
||||
var muxConfig uint32
|
||||
const function = 1 // function 1 is just GPIO
|
||||
|
||||
// SDA
|
||||
muxConfig = function << esp.IO_MUX_GPIO_MCU_SEL_Pos
|
||||
// Make this pin an input pin (always).
|
||||
muxConfig |= esp.IO_MUX_GPIO_FUN_IE
|
||||
// Set drive strength: 0 is lowest, 3 is highest.
|
||||
muxConfig |= 1 << esp.IO_MUX_GPIO_FUN_DRV_Pos
|
||||
config.SDA.mux().Set(muxConfig)
|
||||
config.SDA.outFunc().Set(54)
|
||||
inFunc(54).Set(uint32(esp.GPIO_FUNC_IN_SEL_CFG_SIG_IN_SEL | config.SDA))
|
||||
config.SDA.Set(true)
|
||||
// Configure the pad with the given IO mux configuration.
|
||||
config.SDA.pinReg().SetBits(esp.GPIO_PIN_PIN_PAD_DRIVER)
|
||||
|
||||
esp.GPIO.ENABLE.SetBits(1 << int(config.SDA))
|
||||
esp.I2C.SetCTR_SDA_FORCE_OUT(1)
|
||||
|
||||
// SCL
|
||||
muxConfig = function << esp.IO_MUX_GPIO_MCU_SEL_Pos
|
||||
// Make this pin an input pin (always).
|
||||
muxConfig |= esp.IO_MUX_GPIO_FUN_IE
|
||||
// Set drive strength: 0 is lowest, 3 is highest.
|
||||
muxConfig |= 1 << esp.IO_MUX_GPIO_FUN_DRV_Pos
|
||||
config.SCL.mux().Set(muxConfig)
|
||||
config.SCL.outFunc().Set(53)
|
||||
inFunc(53).Set(uint32(config.SCL))
|
||||
config.SCL.Set(true)
|
||||
// Configure the pad with the given IO mux configuration.
|
||||
config.SCL.pinReg().SetBits(esp.GPIO_PIN_PIN_PAD_DRIVER)
|
||||
|
||||
esp.GPIO.ENABLE.SetBits(1 << int(config.SCL))
|
||||
esp.I2C.SetCTR_SCL_FORCE_OUT(1)
|
||||
}
|
||||
|
||||
// go: inline
|
||||
func (i2c *I2C) initFrequency(config I2CConfig) {
|
||||
|
||||
clkmDiv := i2cClkSourceFrequency/(config.Frequency*1024) + 1
|
||||
sclkFreq := i2cClkSourceFrequency / clkmDiv
|
||||
halfCycle := sclkFreq / config.Frequency / 2
|
||||
//SCL
|
||||
sclLow := halfCycle
|
||||
sclWaitHigh := uint32(0)
|
||||
if config.Frequency > 50000 {
|
||||
sclWaitHigh = halfCycle / 8 // compensate the time when freq > 50K
|
||||
}
|
||||
sclHigh := halfCycle - sclWaitHigh
|
||||
// SDA
|
||||
sdaHold := halfCycle / 4
|
||||
sda_sample := halfCycle / 2
|
||||
setup := halfCycle
|
||||
hold := halfCycle
|
||||
|
||||
esp.I2C.SetSCL_LOW_PERIOD(sclLow - 1)
|
||||
esp.I2C.SetSCL_HIGH_PERIOD(sclHigh)
|
||||
esp.I2C.SetSCL_HIGH_PERIOD_SCL_WAIT_HIGH_PERIOD(25)
|
||||
esp.I2C.SetSCL_RSTART_SETUP_TIME(setup)
|
||||
esp.I2C.SetSCL_STOP_SETUP_TIME(setup)
|
||||
esp.I2C.SetSCL_START_HOLD_TIME(hold - 1)
|
||||
esp.I2C.SetSCL_STOP_HOLD_TIME(hold - 1)
|
||||
esp.I2C.SetSDA_SAMPLE_TIME(sda_sample)
|
||||
esp.I2C.SetSDA_HOLD_TIME(sdaHold)
|
||||
}
|
||||
|
||||
// go: inline
|
||||
func (i2c *I2C) startMaster() {
|
||||
// FIFO mode for data
|
||||
esp.I2C.SetFIFO_CONF_NONFIFO_EN(0)
|
||||
// Reset TX & RX buffers
|
||||
esp.I2C.SetFIFO_CONF_RX_FIFO_RST(1)
|
||||
esp.I2C.SetFIFO_CONF_RX_FIFO_RST(0)
|
||||
esp.I2C.SetFIFO_CONF_TX_FIFO_RST(1)
|
||||
esp.I2C.SetFIFO_CONF_TX_FIFO_RST(0)
|
||||
// set timeout value
|
||||
esp.I2C.TO.Set(0x10)
|
||||
// enable master mode
|
||||
esp.I2C.CTR.Set(0x113)
|
||||
esp.I2C.SetCTR_CONF_UPGATE(1)
|
||||
resetMaster()
|
||||
}
|
||||
|
||||
// go: inline
|
||||
func resetMaster() {
|
||||
// reset FSM
|
||||
esp.I2C.SetCTR_FSM_RST(1)
|
||||
// clear the bus
|
||||
esp.I2C.SetSCL_SP_CONF_SCL_RST_SLV_NUM(9)
|
||||
esp.I2C.SetSCL_SP_CONF_SCL_RST_SLV_EN(1)
|
||||
esp.I2C.SetSCL_STRETCH_CONF_SLAVE_SCL_STRETCH_EN(1)
|
||||
esp.I2C.SetCTR_CONF_UPGATE(1)
|
||||
esp.I2C.FILTER_CFG.Set(0x377)
|
||||
// wait for SCL_RST_SLV_EN
|
||||
for esp.I2C.GetSCL_SP_CONF_SCL_RST_SLV_EN() != 0 {
|
||||
}
|
||||
esp.I2C.SetSCL_SP_CONF_SCL_RST_SLV_NUM(0)
|
||||
}
|
||||
|
||||
type i2cCommandType = uint32
|
||||
type i2cAck = uint32
|
||||
|
||||
const (
|
||||
i2cCMD_RSTART i2cCommandType = 6 << 11
|
||||
i2cCMD_WRITE i2cCommandType = 1<<11 | 1<<8 // WRITE + ack_check_en
|
||||
i2cCMD_READ i2cCommandType = 3<<11 | 1<<8 // READ + ack_check_en
|
||||
i2cCMD_READLAST i2cCommandType = 3<<11 | 5<<8 // READ + ack_check_en + NACK
|
||||
i2cCMD_STOP i2cCommandType = 2 << 11
|
||||
i2cCMD_END i2cCommandType = 4 << 11
|
||||
)
|
||||
|
||||
type i2cCommand struct {
|
||||
cmd i2cCommandType
|
||||
data []byte
|
||||
head int
|
||||
}
|
||||
|
||||
//go:linkname nanotime runtime.nanotime
|
||||
func nanotime() int64
|
||||
|
||||
func (i2c *I2C) transmit(addr uint16, cmd []i2cCommand, timeoutMS int) error {
|
||||
const intMask = esp.I2C_INT_CLR_END_DETECT_INT_CLR_Msk | esp.I2C_INT_CLR_TRANS_COMPLETE_INT_CLR_Msk | esp.I2C_INT_STATUS_TIME_OUT_INT_ST_Msk | esp.I2C_INT_STATUS_NACK_INT_ST_Msk
|
||||
esp.I2C.INT_CLR.SetBits(intMask)
|
||||
esp.I2C.INT_ENA.SetBits(intMask)
|
||||
esp.I2C.SetCTR_CONF_UPGATE(1)
|
||||
|
||||
defer func() {
|
||||
esp.I2C.INT_CLR.SetBits(intMask)
|
||||
esp.I2C.INT_ENA.ClearBits(intMask)
|
||||
}()
|
||||
|
||||
timeoutNS := int64(timeoutMS) * 1000000
|
||||
needAddress := true
|
||||
needRestart := false
|
||||
var readTo []byte
|
||||
for cmdIdx, reg := 0, &esp.I2C.COMD0; cmdIdx < len(cmd); {
|
||||
c := &cmd[cmdIdx]
|
||||
|
||||
switch c.cmd {
|
||||
case i2cCMD_RSTART:
|
||||
reg.Set(i2cCMD_RSTART)
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
cmdIdx++
|
||||
|
||||
case i2cCMD_WRITE:
|
||||
count := 32
|
||||
if needAddress {
|
||||
needAddress = false
|
||||
esp.I2C.SetFIFO_DATA_FIFO_RDATA((uint32(addr) & 0x7f) << 1)
|
||||
count--
|
||||
esp.I2C.SLAVE_ADDR.Set(uint32(addr))
|
||||
esp.I2C.SetCTR_CONF_UPGATE(1)
|
||||
}
|
||||
for ; count > 0 && c.head < len(c.data); count, c.head = count-1, c.head+1 {
|
||||
esp.I2C.SetFIFO_DATA_FIFO_RDATA(uint32(c.data[c.head]))
|
||||
}
|
||||
reg.Set(i2cCMD_WRITE | uint32(32-count))
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
|
||||
if c.head < len(c.data) {
|
||||
reg.Set(i2cCMD_END)
|
||||
reg = nil
|
||||
} else {
|
||||
cmdIdx++
|
||||
}
|
||||
needRestart = true
|
||||
|
||||
case i2cCMD_READ:
|
||||
if needAddress {
|
||||
needAddress = false
|
||||
esp.I2C.SetFIFO_DATA_FIFO_RDATA((uint32(addr)&0x7f)<<1 | 1)
|
||||
esp.I2C.SLAVE_ADDR.Set(uint32(addr))
|
||||
reg.Set(i2cCMD_WRITE | 1)
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
}
|
||||
if needRestart {
|
||||
// We need to send RESTART again after i2cCMD_WRITE.
|
||||
reg.Set(i2cCMD_RSTART)
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
reg.Set(i2cCMD_WRITE | 1)
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
esp.I2C.SetFIFO_DATA_FIFO_RDATA((uint32(addr)&0x7f)<<1 | 1)
|
||||
needRestart = false
|
||||
}
|
||||
count := 32
|
||||
bytes := len(c.data) - c.head
|
||||
// Only last byte in sequence must be sent with ACK set to 1 to indicate end of data.
|
||||
split := bytes <= count
|
||||
if split {
|
||||
bytes--
|
||||
}
|
||||
if bytes > 32 {
|
||||
bytes = 32
|
||||
}
|
||||
reg.Set(i2cCMD_READ | uint32(bytes))
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
|
||||
if split {
|
||||
reg.Set(i2cCMD_READLAST | 1)
|
||||
reg = (*volatile.Register32)(unsafe.Pointer((uintptr(unsafe.Pointer(reg)) + 4)))
|
||||
readTo = c.data[c.head : c.head+bytes+1] // read bytes + 1 last byte
|
||||
cmdIdx++
|
||||
} else {
|
||||
reg.Set(i2cCMD_END)
|
||||
readTo = c.data[c.head : c.head+bytes]
|
||||
reg = nil
|
||||
}
|
||||
|
||||
case i2cCMD_STOP:
|
||||
reg.Set(i2cCMD_STOP)
|
||||
reg = nil
|
||||
cmdIdx++
|
||||
}
|
||||
if reg == nil {
|
||||
// transmit now
|
||||
esp.I2C.SetCTR_CONF_UPGATE(1)
|
||||
esp.I2C.SetCTR_TRANS_START(1)
|
||||
end := nanotime() + timeoutNS
|
||||
var mask uint32
|
||||
for mask = esp.I2C.INT_STATUS.Get(); mask&intMask == 0; mask = esp.I2C.INT_STATUS.Get() {
|
||||
if nanotime() > end {
|
||||
if readTo != nil {
|
||||
return errI2CReadTimeout
|
||||
}
|
||||
return errI2CWriteTimeout
|
||||
}
|
||||
}
|
||||
if mask == esp.I2C_INT_STATUS_TIME_OUT_INT_ST_Msk {
|
||||
if readTo != nil {
|
||||
return errI2CReadTimeout
|
||||
}
|
||||
return errI2CWriteTimeout
|
||||
}
|
||||
esp.I2C.INT_CLR.SetBits(intMask)
|
||||
for i := 0; i < len(readTo); i++ {
|
||||
readTo[i] = byte(esp.I2C.GetFIFO_DATA_FIFO_RDATA() & 0xff)
|
||||
c.head++
|
||||
}
|
||||
readTo = nil
|
||||
reg = &esp.I2C.COMD0
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Tx does a single I2C transaction at the specified address.
|
||||
// It clocks out the given address, writes the bytes in w, reads back len(r)
|
||||
// bytes and stores them in r, and generates a stop condition on the bus.
|
||||
func (i2c *I2C) Tx(addr uint16, w, r []byte) (err error) {
|
||||
// timeout in microseconds.
|
||||
const timeout = 100 * 1000 // 40ms is a reasonable time for a real-time system.
|
||||
|
||||
cmd := make([]i2cCommand, 0, 8)
|
||||
cmd = append(cmd, i2cCommand{cmd: i2cCMD_RSTART})
|
||||
if len(w) > 0 {
|
||||
cmd = append(cmd, i2cCommand{cmd: i2cCMD_WRITE, data: w})
|
||||
}
|
||||
if len(r) > 0 {
|
||||
cmd = append(cmd, i2cCommand{cmd: i2cCMD_READ, data: r})
|
||||
}
|
||||
cmd = append(cmd, i2cCommand{cmd: i2cCMD_STOP})
|
||||
|
||||
return i2c.transmit(addr, cmd, timeout)
|
||||
}
|
||||
|
|
Загрузка…
Создание таблицы
Сослаться в новой задаче