php-linux-gpio/lib/GpioPin.php

259 lines
6.6 KiB
PHP

<?php
/*
* Copyright (C) 2014, NoccyLabs
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>
*/
namespace NoccyLabs\Gpio;
use NoccyLabs\Gpio\Exception\HardwareException;
use NoccyLabs\Sansi\Charset as CS;
/**
* GPIO pin implementation. This class wraps all interaction with a GPIO pin,
* thus decoupling it from the main Gpio class.
*
*/
class GpioPin
{
protected $fd;
protected $pin;
protected $value = 0;
protected $direction;
protected $edge;
protected $handler;
protected $hardware;
protected $label;
protected $gpio;
public function __construct($pin, Gpio $gpio)
{
$this->pin = (int)$pin;
$this->gpio = $gpio;
}
public function getPin()
{
return $this->pin;
}
private function findHardware()
{
$chips = glob("/sys/class/gpio/gpiochip*");
$pinchip = null;
foreach($chips as $chip) {
$r = null;
if (preg_match("/([0-9]+?)/", $chip, $r)) {
$base = (int)$r[1];
if ($base < $this->pin) { $pinchip = $chip; }
}
}
if ($pinchip) {
$hw = trim(file_get_contents($pinchip."/label"));
return $hw;
}
return "unknown";
}
public function __destruct()
{
if ($this->fd) {
fclose($this->fd);
}
}
public function setDirection($direction)
{
if (!in_array($direction, array("in", "out"))) {
throw new \Exception;
}
$this->direction = $direction;
$this->sysfsWrite($this->pin, "direction", $direction);
return $this;
}
public function getDirection()
{
return $this->sysfsRead($this->pin, "direction");
}
public function setValue($value)
{
$this->value = (bool)$value;
@file_put_contents("/sys/class/gpio/gpio{$this->pin}/value", (int)$this->value);
return $this;
}
public function getValue()
{
return $this->sysfsRead($this->pin, "value");
}
/**
* Set a descriptive label for the pin, used for debugging and troubleshooting.
*
* @param string $label
* @return NoccyLabs\Gpio\GpioPin
*/
public function setLabel($label)
{
$this->label = (string)$label;
return $this;
}
/**
* Get the assigned label for this pin.
*
* @return string The label
*/
public function getLabel()
{
return $this->label;
}
public function export()
{
if (file_exists("/sys/class/gpio/gpio{$this->pin}")) {
return $this;
}
$this->sysfsWrite(null, "export", $this->pin);
if (!file_exists("/sys/class/gpio/gpio{$this->pin}")) {
throw new HardwareException("Unable to export pin {$this->pin}");
}
$this->hardware = $this->findHardware();
return $this;
}
public function unexport()
{
$this->sysfsWrite(null, "unexport", $this->pin);
return $this;
}
public function sysfsWrite($pin, $file, $value)
{
if ($pin!==null) {
$path = "/sys/class/gpio/gpio{$pin}/{$file}";
} else {
$path = "/sys/class/gpio/{$file}";
}
if (!file_exists($path)) {
throw new HardwareException("Sysfs file {$path} does not exist");
}
$ts = microtime(true);
while ((!is_writable($path)) && (microtime(true) < $ts+1)) {
usleep(10000);
}
$f = @fopen($path,"c");
if (!is_resource($f)) {
$err = error_get_last();
throw new HardwareException("Unable to write to sysfs file {$path}: ".$err['message']);
}
fwrite($f,$value);
fclose($f);
}
public function sysfsRead($pin, $file)
{
$path = "/sys/class/gpio/gpio{$pin}/{$file}";
$f = fopen($path,"r");
if (!is_resource($f)) {
throw new HardwareException("Unable to read from sysfs file {$path}");
}
$ret = fgets($f);
fclose($f);
return trim($ret);
}
public function setEdge($edge)
{
if (!in_array($edge, array("rising", "falling", "both", "none"))) {
throw new \Exception;
}
$this->edge = $edge;
$this->sysfsWrite($this->pin, "edge", $edge);
return $this;
}
public function getEdge()
{
return $this->sysfsRead($this->pin, "edge");
}
public function setHandler(callable $handler=null)
{
$this->gpio->setInterruptHandler($this->pin, $this->fd);
return $this;
}
public function doInterrupt()
{
fseek($this->fd,0,SEEK_SET);
$val = fgets($this->fd);
}
public function dumpStatus($ansi=false)
{
if ($ansi) {
$status = "\e[44;37;1m GPIO{$this->pin} \e[0m\n";
$direction =
($this->direction=="input"?(CS::chr(0x2190)):(CS::chr(0x2192))).
" ".$this->direction;
$edge = $this->edge; //"";
/*if ($this->edge == "rising") {
$edge.= CS::chr(0x21A5)." rising";
} elseif ($this->edge == "falling") {
$edge.= CS::chr(0x21A7)." falling";
} else {
$edge.= "\e[0m".$this->edge;
}*/
} else {
$status = "********** GPIO{$this->pin} **********\n";
$direction = $this->direction;
}
foreach(array(
"Direction" => $this->getDirection(),
"Value" => ($this->getValue()?1:0),
"Edge" => $this->getEdge(),
"Label" => $this->label,
"Hardware" => $this->hardware,
"Int count" => 0
) as $k=>$v) {
if ($ansi) {
$status.= sprintf(" \e[32;1m%10s\e[0m: \e[0m%s\e[0m\n", $k, $v);
} else {
$status.= sprintf(" %-10s: %s\n", $k, $v);
}
}
error_log($status);
return $this;
}
}