Add configuration for semaphore signals and implement railway signal control

This commit is contained in:
Artem Kashaev
2026-01-22 10:30:31 +05:00
parent 7a455bd6ec
commit 5ae313defd
8 changed files with 638 additions and 111 deletions
+88 -25
View File
@@ -2,35 +2,40 @@ import sys
from switch import Switch
import select
from machine import Pin
from time import sleep_ms
from time import sleep_ms, ticks_diff, ticks_ms
from ir_pair import IRRxTxPollPair
from double_ir import DoubleIr
SEMINSU = dict()
SEMINSU_V1 = dict()
SEMINSU_V2 = dict()
SWITCHES = dict()
LED = Pin("LED", Pin.OUT) # "LED" — специальное имя для встроенного индикатора
def load_switches():
with open("config.json", "r") as file:
import json
config = json.load(file)
for sw_cfg in config["switches"]:
sw = Switch(
id=sw_cfg["id"],
pin=sw_cfg["pin"],
angle_minus=sw_cfg["angle_minus"],
angle_plus=sw_cfg["angle_plus"]
angle_plus=sw_cfg["angle_plus"],
)
SWITCHES[sw.id] = sw
def load_seminsus():
def load_seminsus_v1():
with open("config.json", "r") as file:
import json
config = json.load(file)
for sem_cfg in config["seminsus"]:
if "seminsus_v1" not in config:
return
for sem_cfg in config["seminsus_v1"]:
seminsu = IRRxTxPollPair(
rx_pin=sem_cfg["pin_rx"],
tx_pin=sem_cfg["pin_tx"],
@@ -44,8 +49,20 @@ def load_seminsus():
count_rising=False,
count_falling=True,
)
SEMINSU[sem_cfg["id"]] = seminsu
SEMINSU_V1[sem_cfg["id"]] = seminsu
def load_seminsus_v2():
with open("config.json", "r") as file:
import json
config = json.load(file)
if "irs" not in config:
return
for sem_cfg in config["irs"]:
pin = Pin(sem_cfg["pin"], Pin.IN)
seminsu = DoubleIr(sem_cfg["id"], pin)
SEMINSU_V2[sem_cfg["id"]] = seminsu
def resolve_command(command: str):
@@ -75,7 +92,9 @@ def resolve_command(command: str):
evts = []
for id, sw in SWITCHES.items():
evts.append(f"EVENT SWITCH {id} {sw.pos}")
for id, seminsu in SEMINSU.items():
for id, seminsu in SEMINSU_V1.items():
evts.append(f"EVENT IK_MODULE {id} {seminsu.last_state}")
for id, seminsu in SEMINSU_V2.items():
evts.append(f"EVENT IK_MODULE {id} {seminsu.last_state}")
return "\n".join(evts)
@@ -95,31 +114,65 @@ def _next_seminsu_id():
def poll_seminsus_step():
"""Неблокирующий шаг опроса seminsu.
"""Неблокирующий шаг опроса IK_MODULE (seminsus_v1 + seminsu_v2).
В каждый момент времени опрашивается только ОДНА пара.
Идём round-robin по объединению ID из SEMINSU_V1 и SEMINSU_V2.
ID не пересекаются, поэтому тип выбирается по наличию в словаре.
Для v1 сохраняется текущая неблокирующая логика: start_poll()/update()
и печать события только при изменении состояния.
Для v2 вызывается check() (печать события внутри check() при изменении).
"""
global _active_seminsu_id
if not SEMINSU:
if not SEMINSU_V1 and not SEMINSU_V2:
return
if _active_seminsu_id is None:
_active_seminsu_id = _next_seminsu_id()
if _active_seminsu_id is None:
# Если есть активный v1-цикл — продолжаем его до завершения.
if _active_seminsu_id is not None:
seminsu = SEMINSU_V1.get(_active_seminsu_id)
if seminsu is None:
_active_seminsu_id = None
return
SEMINSU[_active_seminsu_id].start_poll()
seminsu = SEMINSU[_active_seminsu_id]
done = seminsu.update()
if not done:
return
# Цикл завершён — печатаем при изменении состояния.
if seminsu.prev_state is not None and seminsu.last_state is not None:
if seminsu.prev_state != seminsu.last_state:
# state: 1 = перекрыт, 0 = не перекрыт
print(f"EVENT IK_MODULE {_active_seminsu_id} {seminsu.last_state}")
_active_seminsu_id = None
return
# Берём следующий ID из объединённого round-robin.
sid = _next_seminsu_id()
if sid is None:
return
if sid in SEMINSU_V2:
SEMINSU_V2[sid].check()
return
# Иначе — v1.
seminsu = SEMINSU_V1.get(sid)
if seminsu is None:
return
_active_seminsu_id = sid
seminsu.start_poll()
done = seminsu.update()
if not done:
return
# Цикл завершён — печатаем при изменении состояния.
# Быстрый случай: цикл успел завершиться в этом же шаге.
if seminsu.prev_state is not None and seminsu.last_state is not None:
if seminsu.prev_state != seminsu.last_state:
# state: 1 = перекрыт, 0 = не перекрыт
print(f"EVENT IK_MODULE {_active_seminsu_id} {seminsu.last_state}")
print(f"EVENT IK_MODULE {sid} {seminsu.last_state}")
_active_seminsu_id = None
@@ -128,14 +181,22 @@ def work():
poll = select.poll()
poll.register(sys.stdin, select.POLLIN)
# Готовим список id seminsu для round-robin.
# Мигание встроенного светодиода: полный период 1 секунда.
# toggle() каждые 500 мс => 0.5с ON + 0.5с OFF.
LED.value(0)
last_led_toggle_ms = ticks_ms()
# Готовим список id IK_MODULE для round-robin (v1 + v2).
global _seminsu_ids, _seminsu_idx, _active_seminsu_id
_seminsu_ids = list(SEMINSU.keys())
_seminsu_ids = sorted(list(SEMINSU_V1.keys()) + list(SEMINSU_V2.keys()))
_seminsu_idx = 0
_active_seminsu_id = None
while True:
LED.toggle()
now_ms = ticks_ms()
if ticks_diff(now_ms, last_led_toggle_ms) >= 500:
LED.toggle()
last_led_toggle_ms = now_ms
# 1) Обработка stdin НЕ блокирует цикл seminsu.
events = poll.poll(0)
@@ -159,7 +220,9 @@ def work():
# Небольшая пауза, чтобы не крутить CPU на 100%.
sleep_ms(1)
if __name__ == "__main__":
load_switches()
load_seminsus()
work()
load_seminsus_v1()
load_seminsus_v2()
work()