Eljakim Herrewijnen 50b5fc1824 Initial commit
2021-09-27 21:52:27 +02:00

104 lines
3.9 KiB
Lua

local i2c, bit = i2c, bit --luacheck: read globals i2c bit
return function(bus_args)
local busid = bus_args.id or 0
local busad = bus_args.address or 0x27
local speed = bus_args.speed or i2c.SLOW
local rs = bus_args.rs or 0
local rw = bus_args.rw or 1
local en = bus_args.en or 2
local bl = bus_args.backlight or 3
local d4 = bus_args.d4 or 4
local d5 = bus_args.d5 or 5
local d6 = bus_args.d6 or 6
local d7 = bus_args.d7 or 7
-- Convenience I2C setup if a pin configuration is given
if bus_args.sda ~= nil and bus_args.scl ~= nil then
i2c.setup(busid, bus_args.sda, bus_args.scl, speed)
end
-- The onus is on us to maintain the backlight state
local backlight = true
local function send4bitI2C(value, rs_en, rw_en, read)
local function exchange(data, unset_read)
local rv = data
i2c.start(busid)
i2c.address(busid, busad, i2c.TRANSMITTER)
i2c.write(busid, bit.set(data, en))
if read then
i2c.start(busid)
i2c.address(busid, busad, i2c.RECEIVER)
rv = i2c.read(busid, 1):byte(1)
i2c.start(busid)
i2c.address(busid, busad, i2c.TRANSMITTER)
if unset_read then data = bit.bor(bit.bit(rs),
bit.bit(rw),
backlight and bit.bit(bl) or 0) end
i2c.write(busid, bit.set(data, en))
end
i2c.write(busid, bit.clear(data, en))
i2c.stop(busid)
return rv
end
local lo = bit.bor(bit.isset(value, 0) and bit.bit(d4) or 0,
bit.isset(value, 1) and bit.bit(d5) or 0,
bit.isset(value, 2) and bit.bit(d6) or 0,
bit.isset(value, 3) and bit.bit(d7) or 0)
local hi = bit.bor(bit.isset(value, 4) and bit.bit(d4) or 0,
bit.isset(value, 5) and bit.bit(d5) or 0,
bit.isset(value, 6) and bit.bit(d6) or 0,
bit.isset(value, 7) and bit.bit(d7) or 0)
local cmd = bit.bor(rs_en and bit.bit(rs) or 0,
rw_en and bit.bit(rw) or 0,
backlight and bit.bit(bl) or 0)
hi = exchange(bit.bor(cmd, hi), false)
lo = exchange(bit.bor(cmd, lo), true)
return bit.bor(bit.lshift(bit.isset(lo, d4) and 1 or 0, 0),
bit.lshift(bit.isset(lo, d5) and 1 or 0, 1),
bit.lshift(bit.isset(lo, d6) and 1 or 0, 2),
bit.lshift(bit.isset(lo, d7) and 1 or 0, 3),
bit.lshift(bit.isset(hi, d4) and 1 or 0, 4),
bit.lshift(bit.isset(hi, d5) and 1 or 0, 5),
bit.lshift(bit.isset(hi, d6) and 1 or 0, 6),
bit.lshift(bit.isset(hi, d7) and 1 or 0, 7))
end
-- init sequence from datasheet
send4bitI2C(0x33, false, false, false)
send4bitI2C(0x32, false, false, false)
-- Return backend object
return {
fourbits = true,
command = function (_, cmd)
return send4bitI2C(cmd, false, false, false)
end,
busy = function(_)
local rv = send4bitI2C(0xff, false, true, true)
send4bitI2C(bit.bor(0x80, bit.clear(rv, 7)), false, false, false)
return bit.isset(rv, 7)
end,
position = function(_)
local rv = bit.clear(send4bitI2C(0xff, false, true, true), 7)
send4bitI2C(bit.bor(0x80, rv), false, false, false)
return rv
end,
write = function(_, value)
return send4bitI2C(value, true, false, false)
end,
read = function(_)
return send4bitI2C(0xff, true, true, true)
end,
backlight = function(_, on)
backlight = on
local rv = bit.clear(send4bitI2C(0xff, false, true, true), 7)
send4bitI2C(bit.bor(0x80, rv), false, false, false)
return on
end,
}
end