-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathESP8266_video_switch_LM1881_timer.ino
256 lines (225 loc) · 7.97 KB
/
ESP8266_video_switch_LM1881_timer.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
// ===================================================================
//
// Video diversity [WITHOUT RSSI]
//
// GrMis / RCGroups February 2017
//
// seletcs the best video signal, based on the detection
// of line sync. pulses (coming from using LM1881 chpis).
// The souce with the most 'regular' pulses (in the last few hundered lines)
// is chosen.
//
// Hardware:
// - NodeMCU / Wemos D1 (ESP8266) running at 160Mhz
// - Video switch: Max4545
// (allows to commute 4 sources -- but code is here for 3 sources)
// - 3 LM1881, to generate the line sync. pulses
//
// ====================================================================
#include "user_interface.h"
//N is the number of video lines that are averaged over to decide which video source has the highest quality
#define N 500
//Un-comment the line below in order to get some debug prints on the serial output
//#define SERIAL_DEBUG
//Pin connected to the LM1881 sync output for video signal "A"
#define PIN_SyncA D1 //GPIO5=D1
//Pin connected to the LM1881 sync output for video signal "B"
#define PIN_SyncB D2 //GPIO4=D2
//Pin connected to the LM1881 sync output for video signal "C"
#define PIN_SyncC D6
//Pin connected to the MAX4545 (switch) pin that selects the desired source
#define PIN_selectA D3 //GPIO0=D3
#define PIN_selectB D7 //GPIO13=D7
#define PIN_selectC D5 //GPIO14=D5
#define PIN_ESP_LED D0 //GPIO16=D0
#define PIN_ESP_LED2 D4 //GPIO2=D4
// keyword "Volatile" means that this variable may be changed by a routine called by some interrupt.
volatile unsigned long lastA; // Time of the last sync signal
//volatile long periodA;// Period of the last video line, in microseconds
volatile unsigned long lastB; // Time of the last sync signal
//volatile long periodB;// Period of the last video line, in microseconds
volatile unsigned long lastC; // Time of the last sync signal
//volatile long periodC;// Period of the last video line, in microseconds
unsigned long now;
#define t0 63
//const long t0 = 63;
const long unsigned mus = t0 * N; //Period to update the selected source, in micro sec.
const long unsigned mis = mus / 1000; //same as above, in milli sec.
#define sourceA 0
#define sourceB 1
#define sourceC 2
int selected;
unsigned long lastMillis;
//-----------------------------------------------------------------
class ExpAverage {
private:
volatile float x;
const float a, b;
public:
ExpAverage(int n): a(1.0f / n), b(1.0f - a) { //exponential average over the ~n previous values
x = 0.0f;
}
inline float getAverage() {
return x;
}
void addValue(unsigned long v) {
x = a * v + b * x;
}
};
//-----------------------------------------------------------------
ExpAverage A(N), B(N), C(N);//Average video line duration for the different sources
#ifdef SERIAL_DEBUG
ExpAverage AA(N), BB(N), CC(N);//Average video line duration for the different sources
#endif
//-----------------------------------------------------------------
const int CLEAR = (1 << PIN_selectA) + (1 << PIN_selectB) + (1 << PIN_selectC);
int Bin[3] = {1 << PIN_selectA , 1 << PIN_selectB, 1 << PIN_selectC};
inline void SwitchTo(int s) {
if (selected != s) {
selected = s;
GPOC = CLEAR;//Clear PIN_selectA, PIN_selectB and PIN_selectC simultaneously
GPOS = Bin[s];
}
}
//-----------------------------------------------------------------
class PeriodicTask {
private:
os_timer_t myTimer;
int t;
public:
//-----------------------------
static void DoTask(void *pArg) {//static needed here, because below we want to pass a pointer to this member function
const unsigned long now = micros();
if (now - lastA > mus) {
//if no pulse was received since last update, put the value 'mus' into the average
//... so, if no pulse is received at all (disconnected cable, etc.à, the average line duration will be 'mus=N*64' (very large compared to 64) so this
// source will not be selected.
A.addValue(mus); lastA = now;
}
if (now - lastB > mus) B.addValue(mus), lastB = now;
if (now - lastC > mus) C.addValue(mus), lastC = now;
//Get the average deviation (from ~63 micro seconds) of the line durations
const float avA = A.getAverage();
const float avB = B.getAverage();
const float avC = C.getAverage();
if (avB < avA) {
//The best is B or C
if (avC < avB)
{
SwitchTo(sourceC);
//Serial.println(avA); Serial.println(avB); Serial.println(avC);
}
else {
SwitchTo(sourceB);
}
} else {//The best is A or C
if (avA < avC)
SwitchTo(sourceA); else {
SwitchTo(sourceC);
//Serial.println(avA); Serial.println(avB); Serial.println(avC);
}
}
}
//-----------------------------
PeriodicTask () {
os_timer_setfn(&myTimer, DoTask, NULL);
}
void Start(int t, bool repeat = true) {
os_timer_arm(&myTimer, t, repeat);
}
};
PeriodicTask T;
//--------Count---------------------------------------------------------
void CountA() {
const unsigned long now = micros();
const long period = now - lastA;
lastA = now;
A.addValue(abs(period - t0));
#ifdef SERIAL_DEBUG
AA.addValue(period);
#endif
}
void CountB() {
const unsigned long now = micros();
const long period = now - lastB;
lastB = now;
B.addValue(abs(period - t0));
#ifdef SERIAL_DEBUG
BB.addValue(period);
#endif
}
void CountC() {
const unsigned long now = micros();
const long period = now - lastC;
lastC = now;
C.addValue(abs(period - t0));
#ifdef SERIAL_DEBUG
CC.addValue(period);
#endif
}
//-----------------------------------------------------------------
void setup() {
//ESP.wdtDisable(); // To avoid some resets due to the software Watch Dog Timer
//WiFi.mode(WIFI_OFF); WiFi.forceSleepBegin();
#ifdef SERIAL_DEBUG
Serial.begin(115200);
Serial.println("Hello World.");
#endif
lastA = 0; // Time of the last signal on sensor A
lastB = 0; // Time of the last signal on sensor B
lastC = 0; // Time of the last signal on sensor C
pinMode(PIN_SyncA, INPUT_PULLUP);
pinMode(PIN_SyncB, INPUT_PULLUP);
pinMode(PIN_SyncC, INPUT_PULLUP);
pinMode(PIN_selectA, OUTPUT);
pinMode(PIN_selectB, OUTPUT);
pinMode(PIN_selectC, OUTPUT);
pinMode(PIN_ESP_LED, OUTPUT);
selected = sourceB;
//Some blibking to start
for (int i = 0; i < 3; i++) {
SwitchTo(sourceA); delay(250);
SwitchTo(sourceB); delay(250);
SwitchTo(sourceC); delay(250);
}
//Declare 3 interrupts, to measure the duration between horizontal sync. pulses
//on the 3 sources
attachInterrupt(PIN_SyncA, CountA, FALLING);
attachInterrupt(PIN_SyncB, CountB, FALLING);
attachInterrupt(PIN_SyncC, CountC, FALLING);
T.Start(mis, true); //Start the process that updates the selected source every 'mis' milliseconds (timer-based)
}
//-----------------------------------------------------------------
int z = 1;
void loop() {
if (millis() > lastMillis + 1000) {
ESP.wdtFeed();
// Some blinking of the bluit-in led
if (z == 1) digitalWrite(PIN_ESP_LED, HIGH); else digitalWrite(PIN_ESP_LED, LOW); z = -z;
lastMillis = millis();
#ifdef SERIAL_DEBUG
//Prints some debug information to the serial output, every 1000 milli sec.
Serial.print("Time: "); Serial.println(millis());
if (selected == sourceA) Serial.print("=>");
Serial.print("A:");
const float avA = A.getAverage();
Serial.print(avA);
const float avAA = AA.getAverage();
Serial.print(" - "); Serial.println(avAA);
if (selected == sourceB) Serial.print("=>");
Serial.print("B:");
const float avB = B.getAverage();
Serial.print(avB);
const float avBB = BB.getAverage();
Serial.print(" - "); Serial.println(avBB);
if (selected == sourceC) Serial.print("=>");
Serial.print("C:");
const float avC = C.getAverage();
Serial.print(avC);
const float avCC = CC.getAverage();
Serial.print(" - "); Serial.println(avCC);
Serial.println("");
#endif
}
yield();
}