You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
devices/experimental/devices/cap1188/cap1188_test.go

239 lines
7.9 KiB
Go

// Copyright 2017 The Periph Authors. All rights reserved.
// Use of this source code is governed under the Apache License, Version 2.0
// that can be found in the LICENSE file.
package cap1188
import (
"flag"
"io/ioutil"
"log"
"reflect"
"testing"
"time"
"periph.io/x/periph/conn/i2c"
"periph.io/x/periph/conn/i2c/i2ctest"
)
func TestNewI2C(t *testing.T) {
bus := i2ctest.Playback{
Ops: []i2ctest.IO{
// chip ID
{Addr: 40, W: []byte{0xfd}, R: []byte{0x50}},
// clear interrupt
{Addr: 40, W: []byte{0x0}, R: []byte{0x0}},
{Addr: 40, W: []byte{0x0, 0x0}, R: nil},
// enable all inputs
{Addr: 40, W: []byte{0x21, 0xff}, R: nil},
// enable interrupts
{Addr: 40, W: []byte{0x27, 0xff}, R: nil},
// enable/disable repeats
{Addr: 40, W: []byte{0x28, 0xff}, R: nil},
// multitouch
{Addr: 40, W: []byte{0x2a, 0x4}, R: nil},
// sampling
{Addr: 40, W: []byte{0x24, 0x8}, R: nil},
// sensitivity
{Addr: 40, W: []byte{0x1f, 0x50}, R: nil},
// don't retrigger on hold
{Addr: 40, W: []byte{0x28, 0x0}, R: nil},
// config
{Addr: 40, W: []byte{0x20, 0x30}, R: nil},
// config 2
{Addr: 40, W: []byte{0x44, 0x61}, R: nil},
},
}
d, err := NewI2C(&bus, &Opts{Debug: true})
if err != nil {
t.Fatal(err)
}
if s := d.String(); s != "cap1188{playback(40)}" {
t.Fatal(s)
}
if err := d.Halt(); err != nil {
t.Fatal(err)
}
if err := bus.Close(); err != nil {
t.Fatal(err)
}
}
func TestDev_InputStatus(t *testing.T) {
tests := []struct {
name string
bus i2c.Bus
want [8]TouchStatus
}{
{name: "all off",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
// status
{Addr: 40, W: []byte{0x3}, R: []byte{0x0}},
// deltas
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
// thresholds
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus},
},
{name: "all pressed",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0xff}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{PressedStatus, PressedStatus, PressedStatus, PressedStatus, PressedStatus, PressedStatus, PressedStatus, PressedStatus},
},
{name: "first pressed",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0x80}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{PressedStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus},
},
{name: "second pressed",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0x40}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{OffStatus, PressedStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus},
},
{name: "third pressed",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0x20}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{OffStatus, OffStatus, PressedStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus},
},
{name: "eighth pressed",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0x1}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, PressedStatus},
},
{name: "3 pressed",
bus: &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0x91}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
},
want: [8]TouchStatus{PressedStatus, OffStatus, OffStatus, PressedStatus, OffStatus, OffStatus, OffStatus, PressedStatus},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
d, err := NewI2C(tt.bus, DefaultOpts())
if err != nil {
t.Fatal(err)
}
got, err := d.InputStatus()
if err != nil {
t.Fatalf("Dev.InputStatus() error = %v", err)
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("Dev.InputStatus() = %v, want %v", got, tt.want)
}
})
}
// test hold
t.Run("held touch sensors", func(t *testing.T) {
bus := &i2ctest.Playback{
Ops: append(setupPlaybackIO(), []i2ctest.IO{
{Addr: 40, W: []byte{0x3}, R: []byte{0x80}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
// repeat call to get status (still pressed)
{Addr: 40, W: []byte{0x3}, R: []byte{0x80}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
// finall call
{Addr: 40, W: []byte{0x3}, R: []byte{0x0}},
//{Addr: 40, W: []byte{0x10}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
//{Addr: 40, W: []byte{0x30}, R: []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}},
}...),
}
// Set the recorded response to have the retrigger option on.
bus.Ops[10] = i2ctest.IO{Addr: 40, W: []byte{0x28, 0xff}, R: nil}
opts := DefaultOpts()
// Following option needs to be true so we can get the held status.
opts.RetriggerOnHold = true
d, err := NewI2C(bus, opts)
if err != nil {
t.Fatal(err)
}
// first check
got, err := d.InputStatus()
if err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(got, [8]TouchStatus{PressedStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus}) {
t.Fatalf("expected to have the first sensor touched but instead got %v", got)
}
// 2nd check
got, err = d.InputStatus()
if err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(got, [8]TouchStatus{HeldStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus, OffStatus}) {
t.Fatalf("expected to have the first sensor touched but instead got %v", got)
}
})
}
func setupPlaybackIO() []i2ctest.IO {
return []i2ctest.IO{
// chip ID
{Addr: 40, W: []byte{0xfd}, R: []byte{0x50}},
// clear interrupt
{Addr: 40, W: []byte{0x0}, R: []byte{0x0}},
{Addr: 40, W: []byte{0x0, 0x0}, R: nil},
// enable all inputs
{Addr: 40, W: []byte{0x21, 0xff}, R: nil},
// enable interrupts
{Addr: 40, W: []byte{0x27, 0xff}, R: nil},
// enable/disable repeats
{Addr: 40, W: []byte{0x28, 0xff}, R: nil},
// multitouch
{Addr: 40, W: []byte{0x2a, 0x4}, R: nil},
// sampling
{Addr: 40, W: []byte{0x24, 0x8}, R: nil},
// sensitivity
{Addr: 40, W: []byte{0x1f, 0x50}, R: nil},
// linked leds
{Addr: 40, W: []byte{0x72, 0xff}, R: nil},
// don't retrigger on hold
{Addr: 40, W: []byte{0x28, 0x0}, R: nil},
// config
{Addr: 40, W: []byte{0x20, 0x30}, R: nil},
// config 2
{Addr: 40, W: []byte{0x44, 0x61}, R: nil},
}
}
func init() {
sleep = func(time.Duration) {}
flag.Parse()
if !testing.Verbose() {
log.SetOutput(ioutil.Discard)
}
}