betaflight-configurator/js/serial.js

220 lines
8.6 KiB
JavaScript
Raw Normal View History

'use strict';
var serial = {
connectionId: -1,
bitrate: 0,
bytes_received: 0,
bytes_sent: 0,
failed: 0,
2014-03-08 05:25:15 +00:00
transmitting: false,
output_buffer: [],
2014-03-08 05:25:15 +00:00
connect: function (path, options, callback) {
var self = this;
2014-03-08 05:25:15 +00:00
chrome.serial.connect(path, options, function (connectionInfo) {
2014-08-14 14:41:49 +00:00
if (connectionInfo) {
2014-02-03 09:02:49 +00:00
self.connectionId = connectionInfo.connectionId;
self.bitrate = connectionInfo.bitrate;
2014-02-03 09:02:49 +00:00
self.bytes_received = 0;
self.bytes_sent = 0;
self.failed = 0;
2014-03-08 05:25:15 +00:00
2014-02-03 09:02:49 +00:00
self.onReceive.addListener(function log_bytes_received(info) {
self.bytes_received += info.data.byteLength;
});
2014-03-08 05:25:15 +00:00
2014-06-28 12:25:11 +00:00
self.onReceiveError.addListener(function watch_for_on_receive_errors(info) {
console.error(info);
googleAnalytics.sendException('Serial: ' + info.error, false);
2014-06-28 12:25:11 +00:00
switch (info.error) {
case 'system_error': // we might be able to recover from this one
if (!self.failed++) {
chrome.serial.setPaused(self.connectionId, false, function () {
self.getInfo(function (info) {
if (info) {
if (!info.paused) {
console.log('SERIAL: Connection recovered from last onReceiveError');
googleAnalytics.sendException('Serial: onReceiveError - recovered', false);
self.failed = 0;
} else {
console.log('SERIAL: Connection did not recover from last onReceiveError, disconnecting');
GUI.log('Unrecoverable <span style="color: red">failure</span> of serial connection, disconnecting...');
googleAnalytics.sendException('Serial: onReceiveError - unrecoverable', false);
if (GUI.connected_to || GUI.connecting_to) {
$('a.connect').click();
} else {
self.disconnect();
}
}
} else {
if (chrome.runtime.lastError) {
console.error(chrome.runtime.lastError.message);
}
}
});
});
2014-08-09 18:40:40 +00:00
}
break;
case 'timeout':
// TODO
break;
case 'device_lost':
// TODO
break;
case 'disconnected':
// TODO
break;
}
2014-06-28 12:25:11 +00:00
});
console.log('SERIAL: Connection opened with ID: ' + connectionInfo.connectionId + ', Baud: ' + connectionInfo.bitrate);
2014-03-08 05:25:15 +00:00
if (callback) callback(connectionInfo);
2014-02-03 09:02:49 +00:00
} else {
console.log('SERIAL: Failed to open serial port');
googleAnalytics.sendException('Serial: FailedToOpen', false);
if (callback) callback(false);
}
});
},
disconnect: function (callback) {
var self = this;
2014-03-08 05:25:15 +00:00
2014-01-26 23:48:21 +00:00
self.empty_output_buffer();
2014-01-27 00:48:34 +00:00
// remove listeners
2014-01-31 14:51:47 +00:00
for (var i = (self.onReceive.listeners.length - 1); i >= 0; i--) {
self.onReceive.removeListener(self.onReceive.listeners[i]);
2014-01-27 00:48:34 +00:00
}
2014-03-08 05:25:15 +00:00
2014-06-28 12:25:11 +00:00
for (var i = (self.onReceiveError.listeners.length - 1); i >= 0; i--) {
self.onReceiveError.removeListener(self.onReceiveError.listeners[i]);
}
chrome.serial.disconnect(this.connectionId, function (result) {
if (result) {
console.log('SERIAL: Connection with ID: ' + self.connectionId + ' closed');
} else {
console.log('SERIAL: Failed to close connection with ID: ' + self.connectionId + ' closed');
googleAnalytics.sendException('Serial: FailedToClose', false);
}
2014-03-08 05:25:15 +00:00
console.log('SERIAL: Statistics - Sent: ' + self.bytes_sent + ' bytes, Received: ' + self.bytes_received + ' bytes');
2014-03-08 05:25:15 +00:00
self.connectionId = -1;
self.bitrate = 0;
2014-03-08 05:25:15 +00:00
if (callback) callback(result);
});
},
getDevices: function (callback) {
chrome.serial.getDevices(function (devices_array) {
var devices = [];
devices_array.forEach(function (device) {
devices.push(device.path);
});
2014-03-08 05:25:15 +00:00
callback(devices);
});
},
getInfo: function (callback) {
chrome.serial.getInfo(this.connectionId, callback);
},
getControlSignals: function (callback) {
2014-05-02 23:03:24 +00:00
chrome.serial.getControlSignals(this.connectionId, callback);
},
setControlSignals: function (signals, callback) {
2014-07-16 12:06:15 +00:00
chrome.serial.setControlSignals(this.connectionId, signals, callback);
},
send: function (data, callback) {
2014-01-26 19:01:46 +00:00
var self = this;
this.output_buffer.push({'data': data, 'callback': callback});
2014-03-08 05:25:15 +00:00
if (!this.transmitting) {
this.transmitting = true;
2014-03-08 05:25:15 +00:00
var send = function () {
2014-08-09 18:40:40 +00:00
// store inside separate variables in case array gets destroyed
var data = self.output_buffer[0].data,
callback = self.output_buffer[0].callback;
2014-03-08 05:25:15 +00:00
chrome.serial.send(self.connectionId, data, function (sendInfo) {
// track sent bytes for statistics
self.bytes_sent += sendInfo.bytesSent;
// fire callback
2014-08-09 18:40:40 +00:00
callback(sendInfo);
2014-03-08 05:25:15 +00:00
// remove data for current transmission form the buffer
self.output_buffer.shift();
2014-03-08 05:25:15 +00:00
// if there is any data in the queue fire send immediately, otherwise stop trasmitting
2014-08-09 18:40:40 +00:00
if (self.output_buffer.length) {
// keep the buffer withing reasonable limits
if (self.output_buffer.length > 100) {
var counter = 0;
while (self.output_buffer.length > 100) {
self.output_buffer.pop();
counter++;
}
console.log('SERIAL: Send buffer overflowing, dropped: ' + counter + ' entries');
2014-08-09 18:40:40 +00:00
}
2014-03-08 05:25:15 +00:00
send();
2014-08-09 18:40:40 +00:00
} else {
self.transmitting = false;
}
});
};
2014-03-08 05:25:15 +00:00
send();
2014-01-26 19:01:46 +00:00
}
},
2014-01-31 14:51:47 +00:00
onReceive: {
listeners: [],
2014-03-08 05:25:15 +00:00
addListener: function (function_reference) {
chrome.serial.onReceive.addListener(function_reference);
2014-01-31 14:51:47 +00:00
this.listeners.push(function_reference);
},
removeListener: function (function_reference) {
2014-01-31 14:51:47 +00:00
for (var i = (this.listeners.length - 1); i >= 0; i--) {
if (this.listeners[i] == function_reference) {
chrome.serial.onReceive.removeListener(function_reference);
2014-03-08 05:25:15 +00:00
2014-01-31 14:51:47 +00:00
this.listeners.splice(i, 1);
break;
}
2014-03-08 05:25:15 +00:00
}
2014-01-31 14:51:47 +00:00
}
},
2014-06-28 12:25:11 +00:00
onReceiveError: {
listeners: [],
addListener: function (function_reference) {
chrome.serial.onReceiveError.addListener(function_reference);
2014-06-28 12:25:11 +00:00
this.listeners.push(function_reference);
},
removeListener: function (function_reference) {
2014-06-28 12:25:11 +00:00
for (var i = (this.listeners.length - 1); i >= 0; i--) {
if (this.listeners[i] == function_reference) {
chrome.serial.onReceiveError.removeListener(function_reference);
this.listeners.splice(i, 1);
break;
}
}
}
},
empty_output_buffer: function () {
this.output_buffer = [];
this.transmitting = false;
}
};