2014-11-07 16:56:13 -08:00
|
|
|
package sysfs
|
|
|
|
|
|
|
|
import (
|
|
|
|
"syscall"
|
2022-10-26 18:21:34 +02:00
|
|
|
"unsafe"
|
2014-11-07 16:56:13 -08:00
|
|
|
)
|
|
|
|
|
2014-12-31 06:12:25 -08:00
|
|
|
// SystemCaller represents a Syscall
|
2022-10-26 18:21:34 +02:00
|
|
|
// Prevent unsafe call, since go 1.15, see "Pattern 4" in: https://go101.org/article/unsafe.html
|
2014-11-07 16:56:13 -08:00
|
|
|
type SystemCaller interface {
|
2022-10-26 18:21:34 +02:00
|
|
|
Syscall(trap uintptr, f File, signal uintptr, payload unsafe.Pointer) (r1, r2 uintptr, err syscall.Errno)
|
2014-11-07 16:56:13 -08:00
|
|
|
}
|
|
|
|
|
2014-12-31 06:12:25 -08:00
|
|
|
// NativeSyscall represents the native Syscall
|
2014-11-07 16:56:13 -08:00
|
|
|
type NativeSyscall struct{}
|
2014-12-31 06:12:25 -08:00
|
|
|
|
2022-10-26 18:21:34 +02:00
|
|
|
// MockSyscall represents the mock Syscall used for unit tests
|
2017-02-27 23:40:14 +01:00
|
|
|
type MockSyscall struct {
|
2022-10-26 18:21:34 +02:00
|
|
|
lastTrap uintptr
|
|
|
|
lastFile File
|
|
|
|
lastSignal uintptr
|
|
|
|
devAddress uintptr
|
|
|
|
smbus *i2cSmbusIoctlData
|
|
|
|
dataSlice []byte
|
|
|
|
Impl func(trap, a1, a2, a3 uintptr) (r1, r2 uintptr, err syscall.Errno)
|
2017-02-27 23:40:14 +01:00
|
|
|
}
|
2014-11-07 16:56:13 -08:00
|
|
|
|
|
|
|
var sys SystemCaller = &NativeSyscall{}
|
|
|
|
|
2014-12-31 06:12:25 -08:00
|
|
|
// SetSyscall sets the Syscall implementation
|
2014-11-07 16:56:13 -08:00
|
|
|
func SetSyscall(s SystemCaller) {
|
|
|
|
sys = s
|
|
|
|
}
|
|
|
|
|
2014-12-31 06:12:25 -08:00
|
|
|
// Syscall calls either the NativeSyscall or user defined Syscall
|
2022-10-26 18:21:34 +02:00
|
|
|
func Syscall(trap uintptr, f File, a2 uintptr, payload unsafe.Pointer) (r1, r2 uintptr, err syscall.Errno) {
|
|
|
|
return sys.Syscall(trap, f, a2, payload)
|
2014-11-07 16:56:13 -08:00
|
|
|
}
|
|
|
|
|
2022-10-26 18:21:34 +02:00
|
|
|
// Syscall calls the native syscall.Syscall, implements the SystemCaller interface
|
|
|
|
func (sys *NativeSyscall) Syscall(trap uintptr, f File, signal uintptr, payload unsafe.Pointer) (r1, r2 uintptr, err syscall.Errno) {
|
|
|
|
return syscall.Syscall(trap, f.Fd(), signal, uintptr(payload))
|
2014-11-07 16:56:13 -08:00
|
|
|
}
|
|
|
|
|
2022-10-26 18:21:34 +02:00
|
|
|
// Syscall calls the user defined implementation, used for tests, implements the SystemCaller interface
|
|
|
|
func (sys *MockSyscall) Syscall(trap uintptr, f File, signal uintptr, payload unsafe.Pointer) (r1, r2 uintptr, err syscall.Errno) {
|
|
|
|
sys.lastTrap = trap // points to the used syscall (e.g. "SYS_IOCTL")
|
|
|
|
sys.lastFile = f // a character device file (e.g. file to path "/dev/i2c-1")
|
|
|
|
sys.lastSignal = signal // points to used function type (e.g. I2C_SMBUS, I2C_RDWR)
|
|
|
|
|
|
|
|
if signal == I2C_SLAVE {
|
|
|
|
// in this case the uintptr corresponds the address
|
|
|
|
sys.devAddress = uintptr(payload)
|
|
|
|
}
|
|
|
|
|
|
|
|
if signal == I2C_SMBUS {
|
|
|
|
// set the I2C smbus data object reference to payload and fill with some data
|
|
|
|
sys.smbus = (*i2cSmbusIoctlData)(payload)
|
|
|
|
|
|
|
|
// get the data object payload as byte slice
|
|
|
|
if sys.smbus.readWrite == I2C_SMBUS_WRITE {
|
|
|
|
if sys.smbus.data != nil {
|
|
|
|
sys.dataSlice = unsafe.Slice((*byte)(unsafe.Pointer(sys.smbus.data)), sys.smbus.size-1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// fill data object with data from given slice to simulate reading
|
|
|
|
if sys.smbus.readWrite == I2C_SMBUS_READ {
|
|
|
|
if sys.dataSlice != nil {
|
|
|
|
dataSize := sys.smbus.size - 1
|
|
|
|
if sys.smbus.size == I2C_SMBUS_BYTE {
|
|
|
|
dataSize = 1
|
|
|
|
}
|
|
|
|
slc := unsafe.Slice((*byte)(unsafe.Pointer(sys.smbus.data)), dataSize)
|
|
|
|
copy(slc, sys.dataSlice)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// call mock implementation
|
2017-02-27 23:40:14 +01:00
|
|
|
if sys.Impl != nil {
|
2022-10-26 18:21:34 +02:00
|
|
|
return sys.Impl(trap, f.Fd(), signal, uintptr(unsafe.Pointer(payload)))
|
2017-02-27 23:40:14 +01:00
|
|
|
}
|
2017-04-17 09:30:41 +02:00
|
|
|
return 0, 0, 0
|
2014-11-07 16:56:13 -08:00
|
|
|
}
|