209 lines
3.9 KiB
Go
209 lines
3.9 KiB
Go
// SPDX-FileCopyrightText: 2019 Kent Gibson <warthog618@gmail.com>
|
|
//
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package gpiocdev
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/warthog618/go-gpiocdev/uapi"
|
|
"golang.org/x/sys/unix"
|
|
)
|
|
|
|
type watcher struct {
|
|
epfd int
|
|
|
|
// eventfd to signal watcher to shutdown
|
|
donefd int
|
|
|
|
// the handler for detected events
|
|
eh EventHandler
|
|
|
|
// closed once watcher exits
|
|
doneCh chan struct{}
|
|
}
|
|
|
|
func newWatcher(fd int32, eh EventHandler) (w *watcher, err error) {
|
|
var epfd, donefd int
|
|
epfd, err = unix.EpollCreate1(unix.EPOLL_CLOEXEC)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
unix.Close(epfd)
|
|
}
|
|
}()
|
|
donefd, err = unix.Eventfd(0, unix.EFD_CLOEXEC)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
unix.Close(donefd)
|
|
}
|
|
}()
|
|
epv := unix.EpollEvent{Events: unix.EPOLLIN, Fd: int32(donefd)}
|
|
err = unix.EpollCtl(epfd, unix.EPOLL_CTL_ADD, int(donefd), &epv)
|
|
if err != nil {
|
|
return
|
|
}
|
|
epv.Fd = int32(fd)
|
|
err = unix.EpollCtl(epfd, unix.EPOLL_CTL_ADD, int(fd), &epv)
|
|
if err != nil {
|
|
return
|
|
}
|
|
w = &watcher{
|
|
epfd: epfd,
|
|
donefd: donefd,
|
|
eh: eh,
|
|
doneCh: make(chan struct{}),
|
|
}
|
|
go w.watch()
|
|
return
|
|
}
|
|
|
|
func (w *watcher) Close() error {
|
|
unix.Write(w.donefd, []byte{1, 0, 0, 0, 0, 0, 0, 0})
|
|
<-w.doneCh
|
|
unix.Close(w.donefd)
|
|
return nil
|
|
}
|
|
|
|
func (w *watcher) watch() {
|
|
epollEvents := make([]unix.EpollEvent, 2)
|
|
defer close(w.doneCh)
|
|
for {
|
|
n, err := unix.EpollWait(w.epfd, epollEvents[:], -1)
|
|
if err != nil {
|
|
if err == unix.EBADF || err == unix.EINVAL {
|
|
// fd closed so exit
|
|
return
|
|
}
|
|
if err == unix.EINTR {
|
|
continue
|
|
}
|
|
panic(fmt.Sprintf("EpollWait unexpected error: %v", err))
|
|
}
|
|
for i := 0; i < n; i++ {
|
|
ev := epollEvents[i]
|
|
fd := ev.Fd
|
|
if fd == int32(w.donefd) {
|
|
unix.Close(w.epfd)
|
|
return
|
|
}
|
|
evt, err := uapi.ReadLineEvent(uintptr(fd))
|
|
if err != nil {
|
|
continue
|
|
}
|
|
le := LineEvent{
|
|
Offset: int(evt.Offset),
|
|
Timestamp: time.Duration(evt.Timestamp),
|
|
Type: LineEventType(evt.ID),
|
|
Seqno: evt.Seqno,
|
|
LineSeqno: evt.LineSeqno,
|
|
}
|
|
w.eh(le)
|
|
}
|
|
}
|
|
}
|
|
|
|
type watcherV1 struct {
|
|
watcher
|
|
|
|
// fd to offset mapping
|
|
evtfds map[int]int
|
|
}
|
|
|
|
func newWatcherV1(fds map[int]int, eh EventHandler) (w *watcherV1, err error) {
|
|
var epfd, donefd int
|
|
epfd, err = unix.EpollCreate1(unix.EPOLL_CLOEXEC)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
unix.Close(epfd)
|
|
}
|
|
}()
|
|
donefd, err = unix.Eventfd(0, unix.EFD_CLOEXEC)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
unix.Close(donefd)
|
|
}
|
|
}()
|
|
epv := unix.EpollEvent{Events: unix.EPOLLIN, Fd: int32(donefd)}
|
|
err = unix.EpollCtl(epfd, unix.EPOLL_CTL_ADD, int(donefd), &epv)
|
|
if err != nil {
|
|
return
|
|
}
|
|
for fd := range fds {
|
|
epv.Fd = int32(fd)
|
|
err = unix.EpollCtl(epfd, unix.EPOLL_CTL_ADD, fd, &epv)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
w = &watcherV1{
|
|
watcher: watcher{
|
|
epfd: epfd,
|
|
donefd: donefd,
|
|
eh: eh,
|
|
doneCh: make(chan struct{}),
|
|
},
|
|
evtfds: fds,
|
|
}
|
|
go w.watch()
|
|
return
|
|
}
|
|
|
|
func (w *watcherV1) Close() error {
|
|
unix.Write(w.donefd, []byte{1, 0, 0, 0, 0, 0, 0, 0})
|
|
<-w.doneCh
|
|
for fd := range w.evtfds {
|
|
unix.Close(fd)
|
|
}
|
|
unix.Close(w.donefd)
|
|
return nil
|
|
}
|
|
|
|
func (w *watcherV1) watch() {
|
|
epollEvents := make([]unix.EpollEvent, len(w.evtfds))
|
|
defer close(w.doneCh)
|
|
for {
|
|
n, err := unix.EpollWait(w.epfd, epollEvents[:], -1)
|
|
if err != nil {
|
|
if err == unix.EBADF || err == unix.EINVAL {
|
|
// fd closed so exit
|
|
return
|
|
}
|
|
if err == unix.EINTR {
|
|
continue
|
|
}
|
|
panic(fmt.Sprintf("EpollWait unexpected error: %v", err))
|
|
}
|
|
for i := 0; i < n; i++ {
|
|
ev := epollEvents[i]
|
|
fd := ev.Fd
|
|
if fd == int32(w.donefd) {
|
|
unix.Close(w.epfd)
|
|
return
|
|
}
|
|
evt, err := uapi.ReadEvent(uintptr(fd))
|
|
if err != nil {
|
|
continue
|
|
}
|
|
le := LineEvent{
|
|
Offset: w.evtfds[int(fd)],
|
|
Timestamp: time.Duration(evt.Timestamp),
|
|
Type: LineEventType(evt.ID),
|
|
}
|
|
w.eh(le)
|
|
}
|
|
}
|
|
}
|