mirror of https://github.com/periph/devices
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.
243 lines
8.0 KiB
Go
243 lines
8.0 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 cap1xxx
|
|
|
|
import (
|
|
"flag"
|
|
"io/ioutil"
|
|
"log"
|
|
"os"
|
|
"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 != "cap1xxx{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)
|
|
}
|
|
var got [8]TouchStatus
|
|
if err := d.InputStatus(got[:]); 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
|
|
var got [8]TouchStatus
|
|
if err := d.InputStatus(got[:]); 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
|
|
if err = d.InputStatus(got[:]); 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) {}
|
|
}
|
|
|
|
func TestMain(m *testing.M) {
|
|
flag.Parse()
|
|
if !testing.Verbose() {
|
|
log.SetOutput(ioutil.Discard)
|
|
}
|
|
os.Exit(m.Run())
|
|
}
|