Code Monkey home page Code Monkey logo

gpio's Introduction

gpio

Build Status GoDoc Go Report Card License: MIT

GPIO library for the Raspberry Pi.

gpio is a Go library for accessing GPIO pins on the Raspberry Pi.

The library was inspired by and borrows from go-rpio, which is fast but lacks interrupt support, and embd, which supports interrupts, but uses sysfs for read/write and has a far broader scope than I require.

⚠️ Deprecation Warning ⚠️

This library relies on the sysfs GPIO interface which is deprecated in the Linux kernel and is due for removal during 2020. Without sysfs, the watch/interrupt features of this library will no longer work.

The sysfs GPIO interface has been superceded in the Linux kernel by the GPIO character device. The newer API is sufficiently different that reworking this library to use that API is not practical. Instead I have written a new library, gpiod, that provides the same functionality as this library but using the GPIO character device.

There are a couple of downsides to switching to gpiod:

  • The API is quite different, mostly due to the differences in the underlying APIs, so it is not a plugin replacement - you will need to do some code rework.
  • It is also slightly slower for both read and write as all hardware access is now via kernel calls rather than via hardware registers. However, if that is an issue for you then you probably should be writing a kernel device driver for your use case rather than trying to do something in userspace.
  • It requires a recent Linux kernel for full functionality. While the GPIO character device has been around since v4.8, the bias and reconfiguration capabilities required to provide functional equivalence to gpio were only added in v5.5.

There are several benefits in the switch:

  • gpiod is not Raspberry Pi specific, but will work on any platform where the GPIO chip is supported by the Linux kernel, so code using gpiod is more portable.
  • gpio writes directly to hardware registers so it can conflict with other kernel drivers. The gpiod accesses the hardware internally using the same interfaces as other kernel drivers and so should play nice with them.
  • gpiod supports Linux GPIO line labels, so you can find your line by name (assuming it has been named by device-tree).
  • and of course, it will continue to work beyond 2020.

I've already ported all my projects that were using gpio to gpiod and strongly suggest that you do the same.

Features

Supports the following functionality:

  • Pin Mode/Direction (Input / Output)
  • Write (High / Low)
  • Read (High / Low)
  • Pullups (Up / Down / None)
  • Watches/Interrupts (Rising/Falling/Both)

Usage

import "github.com/warthog618/gpio"

Library Initialization

Open memory range for GPIO access in /dev/gpiomem

err := gpio.Open()

Cleanup when done

gpio.Close()

Pin Initialization

A Pin object is constructed using the NewPin function. The Pin object is then used for all operations on that pin. Note that the pin number refers to the BCM GPIO pin, not the physical pin on the Raspberry Pi header. Pin 4 here is exposed on the pin header as physical pin 7 (J8 7). Mappings are provided from Raspberry Pi J8 header pin names to BCM GPIO numbers, using the form J8pX.

pin := gpio.NewPin(4)
pin := gpio.NewPin(gpio.J8p7) // Using Raspberry Pi J8 mapping.

There is no need to cleanup a pin if you no longer need to use it, unless it has Watches set in which case you should remove the Watch.

Mode

The pin mode controls whether the pin is an input or output. The existing mode can be read back.

mode := pin.Mode()
pin.Output()               // Set mode to Output
pin.Input()                // Set mode to Input
pin.SetMode(gpio.Output)   // Alternate syntax

To prevent output glitches, the pin level can be set using High/Low/Write before the pin is set to Output.

Input

res := pin.Read()  // Read state from pin (High / Low)

Output

pin.High()              // Set pin High
pin.Low()               // Set pin Low
pin.Toggle()            // Toggle pin (Low -> High -> Low)

pin.Write(gpio.High)    // Alternate syntax

Also see example example/blinker/blinker.go

Pullups

Pull up state can be set using:

pin.PullUp()
pin.PullDown()
pin.PullNone()

pin.SetPull(gpio.PullUp)  // Alternate syntax

Unlike the Mode, the pull up state cannot be read back from hardware, so there is no Pull function.

Watches

The state of an input pin can be watched and trigger calls to handler functions.

The watch can be on rising or falling edges, or both.

The handler function is passed the triggering pin.

func handler(*Pin) {
  // handle change in pin value
}
pin.Watch(gpio.EdgeFalling,handler)    // Call handler when pin changes from High to Low.

pin.Watch(gpio.EdgeRising,handler)     // Call handler when pin changes from Low to High.

pin.Watch(gpio.EdgeBoth,handler)       // Call handler when pin changes

A watch can be removed using the Unwatch function.

pin.Unwatch()

Tools

A command line utility, gppiio, is provided to allow manual and scripted control of GPIO pins:

$ ./gppiio
gppiio is a utility to control Raspberry Pi GPIO pins

Usage:
  gppiio [flags]
  gppiio [command]

Available Commands:
  detect      Identify the GPIO chip
  get         Read the level of a pin or pins
  help        Help about any command
  mode        Read the functional mode of a pin or pins
  mon         Monitor the level of a pin or pins
  pull        Set the pull direction of a pin or pins
  set         Set the level of a pin or pins
  version     Display the version

Flags:
  -h, --help      help for gppiio

Use "gppiio [command] --help" for more information about a command.

Examples

Refer to the examples for more examples of usage.

Examples can be cross-compiled from other platforms using

GOOS=linux GOARCH=arm GOARM=6 go build

Tests

The library is fully tested, other than some error cases that are difficult to test.

The tests are intended to be run on a Raspberry Pi with J8 pin 7 floating and with pins 15 and 16 tied together, possibly using a jumper across the header. The tests set J8 pin 16 to an output so DO NOT run them on hardware where that pin is being externally driven.

Tests have been run successfully on Raspberry Pi B (Rev 1 and Rev 2), B+, Pi2 B, Pi4 B, and Pi Zero W. The library should also work on other Raspberry Pi variants, I just don't have any available to test.

The tests can be cross-compiled from other platforms using

GOOS=linux GOARCH=arm GOARM=6 go test -c

Later Pis can also use ARM7 (GOARM=7).

Benchmarks

The tests include benchmarks on reads and writes. Reading pin levels through sysfs is provided for comparison.

These are the results from a Raspberry Pi Zero W built with Go 1.13:

$ ./gpio.test -test.bench=.*
goos: linux
goarch: arm
pkg: github.com/warthog618/gpio
BenchmarkRead                  9485052           124 ns/op
BenchmarkWrite                18478959          58.8 ns/op
BenchmarkToggle               16695492          72.4 ns/op
BenchmarkInterruptLatency         2348        453248 ns/op
BenchmarkSysfsRead               32983         31004 ns/op
BenchmarkSysfsWrite              17192         69840 ns/op
BenchmarkSysfsToggle             17341         62962 ns/op

PASS

Prerequisites

The library assumes Linux, and has been tested on Raspbian Jessie, Stretch and Buster.

The library targets all models of the Raspberry Pi, upt to and including the Pi 4B. Note that the Raspberry Pi Model B Rev 1.0 has different pinouts, so the J8 mappings are incorrect for that particular revision.

This library utilizes /dev/gpiomem, which must be available to the current user. This is generally available in recent Raspian releases.

The library also utilizes the sysfs GPIO to support interrupts on changes to input pin values. The sysfs is not used to access the pin values, as the gpiomem approach is orders of magnitude faster (refer to the benchmarks).

gpio's People

Contributors

johnsto avatar warthog618 avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar

gpio's Issues

compile err on GO1.11

./dio.go:186:19: invalid operation: 1 << (pin & 31) (shift count type int, must be unsigned integer)
./dio.go:242:28: invalid operation: mem[pin.fsel] >> modeShift (shift count type int, must be unsigned integer)
./dio.go:272:33: invalid operation: modeMask << modeShift (shift count type int, must be unsigned integer)
./dio.go:272:69: invalid operation: uint32(mode) << modeShift (shift count type int, must be unsigned integer)
./dio.go:334:47: invalid operation: pullMask << shift (shift count type int, must be unsigned integer)
./dio.go:334:79: invalid operation: uint32(pull) << shift (shift count type int, must be unsigned integer)

gpio.test test failures on more recent platforms

run gpop.test sometimes OK, sometimes may FAIL.

error message is following:

pi@raspberrypi:~ $ ./gpio.test -test.bench=.*
--- FAIL: TestCloseInterrupts (0.06s)
interrupt_test.go:312: Spurious interrupt during close 0
FAIL

Excellent, thank you.

Just dropping by to say thank you for an excellent library. Source code is clear and idiomatic from a targetted Go/Linux/RPI system perspective. The README gives a nice introduction and justification.

I refactored my code from @hybridgroup's Gobot GPIO ButtonDriver to use your library instead and saw a significant reduction in resource consumption.

Below stats from top are enough to see the huge difference during a non-eventful watch period.

Before:

RES    SHR  S  %CPU  %MEM
7432   2416 S   4.7   1.7

After:

RES    SHR  S  %CPU  %MEM
2404   1964 S   0.0   0.5

I actually had to hunt for the pid because the process no longer shows up in top when ordered by CPU usage. Nice work!

too many GO modules depends

I found it depends on soo many modules which I can't download all, and I can't get it compiled.

after remove file go.mod, go.sum, fallback to go get mode, it compile OK.

maybe somethin error whit go module ?

BTW: I very this project : wonderful function, clear APIs ...

Getting the pin number in the watcher handler

Currently the Pin struct doesn't export a way of getting the pin number that caused to watcher handler to run.

This makes writing generic handlers impossible.

Adding a reader to the Pin struct for the internal pin value would make things way easier. Something like

func (pin *Pin) Pin() uint8 {
  return pin.pin
}

Another option would be to export the pin field, but then it would be possible to set that value externally.

Port to Banana PI M4 Board

Hi Kent,
We are looking for a Golang package to access GPIO on our Banana PI M4 board. Your project seems promising but realize that this is for Raspberry PI board. If we want to use your project for our Banana PI board, what changes do we have to make and where the changes have to be made. Or if you are able to make the changes to it so as to enable support for Banana PI M4 that would be excellent.
Thank you for your help.
David.

how to recover from a crash?

hi,
my app crashed for the first time while using this library (nil pointer dereference) so it couldn't call Unwatch() on all the buttons nor gpio.Close().
when the app is now started again, I get the error that the pin is already in use.
what is the proper way to claim a pin again? I actually want to "force use" the pin because the application must always "just work".

panic: pin already in use

no error on first run. when I cancel the program with ctrl-c getting this error. (Raspberry pi 3 B+)


err=pin.Watch(gpio.EdgeBoth, func(pin *gpio.Pin) {
fmt.Printf("Pin 4 is %v\n", pin.Read())
})
if err!=nil {
panic(err)
}

defer pin.Unwatch()

panic: pin already in use

goroutine 1 [running]:
panic(0xd2b10, 0x1040c198)
/usr/lib/go-1.7/src/runtime/panic.go:500 +0x33c

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.