1
0
Fork 0
mirror of https://github.com/iNavFlight/inav-configurator.git synced 2025-07-13 11:29:53 +03:00
inav-configurator/js/serial_queue.js
2017-01-21 13:29:41 +01:00

270 lines
No EOL
7.5 KiB
JavaScript

'use strict';
var helper = helper || {};
var SimpleSmoothFilterClass = function (initialValue, smoothingFactor) {
var publicScope = {};
publicScope.value = initialValue;
publicScope.smoothFactor = smoothingFactor;
if (publicScope.smoothFactor >= 1) {
publicScope.smoothFactor = 0.99;
}
if (publicScope.smoothFactor <= 0) {
publicScope.smoothFactor = 0;
}
publicScope.apply = function (newValue) {
publicScope.value = (newValue * (1 - publicScope.smoothFactor)) + (publicScope.value * publicScope.smoothFactor);
return publicScope;
};
publicScope.get = function () {
return publicScope.value;
};
return publicScope;
};
//FIXME extract it to separate file
var WalkingAverageClass = function (maxLength) {
var table = [],
self = {};
/**
*
* @param {number} data
*/
self.put = function (data) {
table.push(data);
if (table.length > maxLength) {
table.shift();
}
};
self.getAverage = function () {
if (table.length > 0) {
var sum = table.reduce(function (a, b) {
return a + b;
});
return sum / table.length;
} else {
return 0;
}
};
return self;
};
helper.mspQueue = (function (serial, MSP) {
var publicScope = {},
privateScope = {};
privateScope.handlerFrequency = 100;
privateScope.balancerFrequency = 10;
privateScope.loadAverage = new WalkingAverageClass(privateScope.handlerFrequency);
privateScope.roundtripAverage = new WalkingAverageClass(50);
privateScope.hardwareRoundtripAverage = new WalkingAverageClass(50);
privateScope.pastLoadFilter = new SimpleSmoothFilterClass(1, 0.99);
privateScope.currentLoadFilter = new SimpleSmoothFilterClass(1, 0.7);
privateScope.targetLoad = 1.5;
privateScope.currentLoad = 0;
privateScope.loadPid = {
gains: {
P: 10,
I: 4,
D: 2
},
Iterm: 0,
ItermLimit: 80,
previousError: 0,
output: {
min: 0,
max: 95,
minThreshold: 2
}
};
privateScope.dropRatio = 0;
publicScope.computeDropRatio = function () {
var error = privateScope.currentLoad - privateScope.targetLoad;
var Pterm = error * privateScope.loadPid.gains.P,
Dterm = (error - privateScope.loadPid.previousError) * privateScope.loadPid.gains.P;
privateScope.loadPid.previousError = error;
privateScope.loadPid.Iterm += error * privateScope.loadPid.gains.I;
if (privateScope.loadPid.Iterm > privateScope.loadPid.ItermLimit) {
privateScope.loadPid.Iterm = privateScope.loadPid.ItermLimit;
} else if (privateScope.loadPid.Iterm < -privateScope.loadPid.ItermLimit) {
privateScope.loadPid.Iterm = -privateScope.loadPid.ItermLimit;
}
privateScope.dropRatio = Pterm + privateScope.loadPid.Iterm + Dterm;
if (privateScope.dropRatio < privateScope.loadPid.output.minThreshold) {
privateScope.dropRatio = privateScope.loadPid.output.min;
}
if (privateScope.dropRatio > privateScope.loadPid.output.max) {
privateScope.dropRatio = privateScope.loadPid.output.max;
}
};
publicScope.getDropRatio = function () {
return privateScope.dropRatio;
};
privateScope.queue = [];
privateScope.portInUse = false;
/**
* This method is periodically executed and moves MSP request
* from a queue to serial port. This allows to throttle requests,
* adjust rate of new frames being sent and prohibit situation in which
* serial port is saturated, virtually overloaded, with outgoing data
*
* This also implements serial port sharing problem: only 1 frame can be transmitted
* at once
*
* MSP class no longer implements blocking, it is queue responsibility
*/
publicScope.executor = function () {
privateScope.loadAverage.put(privateScope.queue.length);
privateScope.pastLoadFilter.apply(privateScope.currentLoad);
privateScope.currentLoadFilter.apply(privateScope.currentLoad);
/*
* if port is blocked or there is no connection, do not process the queue
*/
if (privateScope.portInUse || serial.connectionId === false) {
return false;
}
var request = privateScope.get();
if (request !== undefined) {
/*
* Lock serial port as being in use right now
*/
privateScope.portInUse = true;
request.timer = setTimeout(function () {
console.log('MSP data request timed-out: ' + request.code);
/*
* Remove current callback
*/
MSP.removeCallback(request.code);
/*
* Create new entry in the queue
*/
publicScope.put(request);
}, serial.getTimeout());
if (request.sentOn === null) {
request.sentOn = new Date().getTime();
}
/*
* Set receive callback here
*/
MSP.putCallback(request);
/*
* Send data to serial port
*/
serial.send(request.messageBody, function (sendInfo) {
if (sendInfo.bytesSent == request.messageBody.byteLength) {
/*
* message has been sent, check callbacks and free resource
*/
if (request.onSend) {
request.onSend();
}
privateScope.portInUse = false;
}
});
}
};
privateScope.get = function () {
return privateScope.queue.shift();
};
publicScope.flush = function () {
privateScope.queue = [];
};
publicScope.freeSerialPort = function () {
privateScope.portInUse = false;
};
publicScope.put = function (mspRequest) {
privateScope.queue.push(mspRequest);
};
publicScope.getLength = function () {
return privateScope.queue.length;
};
/**
* 1s MSP load computed as number of messages in a queue in given period
* @returns {number}
*/
publicScope.getLoad = function () {
return privateScope.currentLoad;
};
publicScope.getRoundtrip = function () {
return privateScope.roundtripAverage.getAverage();
};
/**
*
* @param {number} number
*/
publicScope.putRoundtrip = function (number) {
privateScope.roundtripAverage.put(number);
};
publicScope.getHardwareRoundtrip = function () {
return privateScope.hardwareRoundtripAverage.getAverage();
};
/**
*
* @param {number} number
*/
publicScope.putHardwareRoundtrip = function (number) {
privateScope.hardwareRoundtripAverage.put(number);
};
publicScope.balancer = function () {
privateScope.currentLoad = privateScope.loadAverage.getAverage();
helper.mspQueue.computeDropRatio();
};
publicScope.shouldDrop = function () {
return (Math.round(Math.random()*100) < privateScope.dropRatio);
};
setInterval(publicScope.executor, Math.round(1000 / privateScope.handlerFrequency));
setInterval(publicScope.balancer, Math.round(1000 / privateScope.balancerFrequency));
return publicScope;
})(serial, MSP);