cylon/lib/digital-pin.js

198 lines
5.4 KiB
JavaScript
Raw Normal View History

/*
* Linux IO DigitalPin
* cylonjs.com
*
2014-02-28 03:28:13 +08:00
* Copyright (c) 2013-2014 The Hybrid Group
* Licensed under the Apache 2.0 license.
*/
2014-02-28 03:28:13 +08:00
'use strict';
var FS = require('fs');
var EventEmitter = require('events').EventEmitter;
var namespace = require('node-namespace');
2014-02-28 06:42:33 +08:00
// DigitalPin class offers an interface with the Linux GPIO system present in
// single-board computers such as a Raspberry Pi, or a BeagleBone
2014-02-28 03:28:13 +08:00
namespace("Cylon.IO", function() {
this.DigitalPin = (function(klass) {
subclass(DigitalPin, klass);
var GPIO_PATH = "/sys/class/gpio";
var GPIO_READ = "in";
var GPIO_WRITE = "out";
var HIGH = 1;
var LOW = 0;
function DigitalPin(opts) {
this.pinNum = opts.pin;
this.status = 'low';
this.ready = false;
this.mode = opts.mode;
}
DigitalPin.prototype.connect = function(mode) {
var _this = this;
if (mode == null) {
mode = null;
}
2014-02-28 03:28:13 +08:00
if (this.mode == null) {
this.mode = mode;
}
return FS.exists(this._pinPath(), function(exists) {
if (exists) {
return _this._openPin();
} else {
return _this._createGPIOPin();
}
2014-02-28 03:28:13 +08:00
});
};
DigitalPin.prototype.close = function() {
var _this = this;
return FS.writeFile(this._unexportPath(), "" + this.pinNum, function(err) {
return _this._closeCallback(err);
});
};
DigitalPin.prototype.closeSync = function() {
FS.writeFileSync(this._unexportPath(), "" + this.pinNum);
return this._closeCallback(false);
};
DigitalPin.prototype.digitalWrite = function(value) {
var _this = this;
if (this.mode !== 'w') {
this._setMode('w');
}
this.status = value === 1 ? 'high' : 'low';
FS.writeFile(this._valuePath(), value, function(err) {
if (err) {
return _this.emit('error', "Error occurred while writing value " + value + " to pin " + _this.pinNum);
2013-11-13 01:47:25 +08:00
} else {
2014-02-28 03:28:13 +08:00
return _this.emit('digitalWrite', value);
2013-11-13 01:47:25 +08:00
}
2014-02-28 03:28:13 +08:00
});
return value;
};
2014-02-28 06:42:33 +08:00
// Public: Reads the digial pin's value periodicly on a supplied interval,
// and emits the result or an error
//
// interval - time (in milliseconds) to read from the pin at
//
// Returns the defined interval
2014-02-28 03:28:13 +08:00
DigitalPin.prototype.digitalRead = function(interval) {
var readData,
_this = this;
if (this.mode !== 'r') {
this._setMode('r');
}
readData = null;
return every(interval, function() {
return FS.readFile(_this._valuePath(), function(err, data) {
2013-11-13 01:47:25 +08:00
if (err) {
2014-02-28 03:28:13 +08:00
return _this.emit('error', "Error occurred while reading from pin " + _this.pinNum);
2013-11-13 01:47:25 +08:00
} else {
2014-02-28 03:28:13 +08:00
readData = parseInt(data.toString());
return _this.emit('digitalRead', readData);
2013-11-13 01:47:25 +08:00
}
});
2014-02-28 03:28:13 +08:00
});
};
DigitalPin.prototype.setHigh = function() {
return this.digitalWrite(1);
};
DigitalPin.prototype.setLow = function() {
return this.digitalWrite(0);
};
DigitalPin.prototype.toggle = function() {
if (this.status === 'low') {
return this.setHigh();
} else {
return this.setLow();
}
};
2013-11-13 01:47:25 +08:00
2014-02-28 06:42:33 +08:00
// Creates the GPIO file to read/write from
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._createGPIOPin = function() {
var _this = this;
return FS.writeFile(this._exportPath(), "" + this.pinNum, function(err) {
2013-11-13 01:47:25 +08:00
if (err) {
2014-02-28 03:28:13 +08:00
return _this.emit('error', 'Error while creating pin files');
2013-11-13 01:47:25 +08:00
} else {
2014-02-28 03:28:13 +08:00
return _this._openPin();
2013-11-13 01:47:25 +08:00
}
2014-02-28 03:28:13 +08:00
});
};
DigitalPin.prototype._openPin = function() {
this._setMode(this.mode, true);
return this.emit('open');
};
DigitalPin.prototype._closeCallback = function(err) {
if (err) {
return this.emit('error', 'Error while closing pin files');
} else {
return this.emit('close', this.pinNum);
}
};
2013-11-12 03:44:48 +08:00
2014-02-28 06:42:33 +08:00
// Sets the mode for the GPIO pin by writing the correct values to the pin reference files
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._setMode = function(mode, emitConnect) {
var _this = this;
if (emitConnect == null) {
emitConnect = false;
}
this.mode = mode;
if (mode === 'w') {
return FS.writeFile(this._directionPath(), GPIO_WRITE, function(err) {
return _this._setModeCallback(err, emitConnect);
});
} else if (mode === 'r') {
return FS.writeFile(this._directionPath(), GPIO_READ, function(err) {
return _this._setModeCallback(err, emitConnect);
});
}
};
DigitalPin.prototype._setModeCallback = function(err, emitConnect) {
if (err) {
return this.emit('error', "Setting up pin direction failed");
} else {
this.ready = true;
if (emitConnect) {
return this.emit('connect', this.mode);
2013-11-13 01:47:25 +08:00
}
2014-02-28 03:28:13 +08:00
}
};
2013-11-12 03:44:48 +08:00
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._directionPath = function() {
return "" + (this._pinPath()) + "/direction";
};
2013-11-12 03:44:48 +08:00
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._valuePath = function() {
return "" + (this._pinPath()) + "/value";
};
2013-11-12 03:44:48 +08:00
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._pinPath = function() {
return "" + GPIO_PATH + "/gpio" + this.pinNum;
};
2013-11-12 03:44:48 +08:00
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._exportPath = function() {
return "" + GPIO_PATH + "/export";
};
2014-02-28 03:28:13 +08:00
DigitalPin.prototype._unexportPath = function() {
return "" + GPIO_PATH + "/unexport";
};
2014-02-28 03:28:13 +08:00
return DigitalPin;
2014-02-28 03:28:13 +08:00
})(EventEmitter);
});