hybridgroup.gobot/system/syscall_mock.go

84 lines
2.4 KiB
Go

package system
import (
"unsafe"
)
// mockSyscall represents the mock Syscall used for unit tests
type mockSyscall struct {
lastTrap uintptr
lastFile File
lastSignal uintptr
devAddress uintptr
smbus *i2cSmbusIoctlData
sliceSize uint8
dataSlice []byte
Impl func(trap, a1, a2 uintptr, a3 unsafe.Pointer) (r1, r2 uintptr, err SyscallErrno)
}
// Syscall calls the user defined implementation, used for tests, implements the SystemCaller interface
//
//nolint:nonamedreturns // useful here
func (sys *mockSyscall) syscall(
trap uintptr,
f File,
signal uintptr,
payload unsafe.Pointer,
address uint16,
) (r1, r2 uintptr, err SyscallErrno) {
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 {
// this is the setup for the address, it needs to be converted to an uintptr,
// the given payload is not used in this case, see the comment on the function used for production
sys.devAddress = uintptr(address)
}
if signal == I2C_SMBUS {
// set the I2C smbus data object reference to payload and fill with some data
sys.smbus = (*i2cSmbusIoctlData)(payload)
if sys.smbus.data != nil {
sys.sliceSize = sys.retrieveSliceSize()
if sys.smbus.readWrite == I2C_SMBUS_WRITE {
// get the data object payload as byte slice
sys.dataSlice = unsafe.Slice((*byte)(sys.smbus.data), sys.sliceSize)
}
if sys.smbus.readWrite == I2C_SMBUS_READ {
// fill data object with data from given slice to simulate reading
if sys.dataSlice != nil {
slc := unsafe.Slice((*byte)(sys.smbus.data), sys.sliceSize)
if sys.smbus.protocol == I2C_SMBUS_BLOCK_DATA || sys.smbus.protocol == I2C_SMBUS_I2C_BLOCK_DATA {
copy(slc[1:], sys.dataSlice)
} else {
copy(slc, sys.dataSlice)
}
}
}
}
}
// call mock implementation
if sys.Impl != nil {
return sys.Impl(trap, f.Fd(), signal, payload)
}
return 0, 0, 0
}
func (sys *mockSyscall) retrieveSliceSize() uint8 {
switch sys.smbus.protocol {
case I2C_SMBUS_BYTE:
return 1
case I2C_SMBUS_BYTE_DATA:
return 1
case I2C_SMBUS_WORD_DATA:
return 2
default:
// for I2C_SMBUS_BLOCK_DATA, I2C_SMBUS_I2C_BLOCK_DATA
return *(*byte)(sys.smbus.data) + 1 // first data element contains data size
}
}