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
|
-- train_related.lua
-- Occupation of track sections - mainly implementation of train callbacks
--[[
Track section occupation is saved as follows
In train:
train.il_sections = {
[n] = {ts_id = <...> (origin = <sigd>)}
}
-- "origin" is the TCB (signal describer) the train initially entered this section
In track section
ts.trains = {
[n] = <train_id>
}
When any inconsistency is detected, we will assume the most restrictive setup.
It will be possible to indicate a section "free" via the GUI.
]]
local ildb = advtrains.interlocking.db
local sigd_equal = advtrains.interlocking.sigd_equal
local function itexist(tbl, com)
for _,item in ipairs(tbl) do
if (item==com) then
return true
end
end
return false
end
local function itkexist(tbl, ikey, com)
for _,item in ipairs(tbl) do
if item[ikey] == com then
return true
end
end
return false
end
local function itremove(tbl, com)
local i=1
while i <= #tbl do
if tbl[i] == com then
table.remove(tbl, i)
else
i = i + 1
end
end
end
local function itkremove(tbl, ikey, com)
local i=1
while i <= #tbl do
if tbl[i][ikey] == com then
table.remove(tbl, i)
else
i = i + 1
end
end
end
local function setsection(tid, train, ts_id, ts, sigd)
-- train
if not train.il_sections then train.il_sections = {} end
if not itkexist(train.il_sections, "ts_id", ts_id) then
table.insert(train.il_sections, {ts_id = ts_id, origin = origin})
end
-- ts
if not ts.trains then ts.trains = {} end
if not itexist(ts.trains, tid) then
table.insert(ts.trains, tid)
end
-- routes
local tcbs = advtrains.interlocking.db.get_tcbs(sigd)
-- route setting - clear route state
if ts.route then
--atdebug(tid,"enters",ts_id,"examining Routestate",ts.route)
if not sigd_equal(ts.route.entry, sigd) then
-- Train entered not from the route. Locate origin and cancel route!
atwarn("Train",tid,"hit route",ts.route.rsn,"!")
advtrains.interlocking.route.cancel_route_from(ts.route.origin)
atwarn("Route was cancelled.")
end
-- train entered route regularily. Reset route and signal
tcbs.route_committed = nil
tcbs.route_comitted = nil -- TODO compatibility cleanup
tcbs.aspect = nil
tcbs.route_origin = nil
advtrains.interlocking.update_signal_aspect(tcbs)
if tcbs.signal and sigd_equal(ts.route.entry, ts.route.origin) then
if tcbs.route_auto and tcbs.routeset then
--atdebug("Resetting route (",ts.route.origin,")")
advtrains.interlocking.route.update_route(ts.route.origin, tcbs)
else
tcbs.routeset = nil
end
end
ts.route = nil
end
if tcbs.signal then
advtrains.interlocking.route.update_route(sigd, tcbs)
end
end
local function freesection(tid, train, ts_id, ts)
-- train
if not train.il_sections then train.il_sections = {} end
itkremove(train.il_sections, "ts_id", ts_id)
-- ts
if not ts.trains then ts.trains = {} end
itremove(ts.trains, tid)
if ts.route_post then
advtrains.interlocking.route.free_route_locks(ts_id, ts.route_post.locks)
if ts.route_post.next then
--this does nothing when the train went the right way, because
-- "route" info is already cleared.
advtrains.interlocking.route.cancel_route_from(ts.route_post.next)
end
ts.route_post = nil
end
-- This must be delayed, because this code is executed in-between a train step
-- TODO use luaautomation timers?
minetest.after(0, advtrains.interlocking.route.update_waiting, "ts", ts_id)
end
-- This is regular operation
-- The train is on a track and drives back and forth
-- This sets the section for both directions, to be failsafe
advtrains.tnc_register_on_enter(function(pos, id, train, index)
local tcb = ildb.get_tcb(pos)
if tcb then
for connid=1,2 do
local ts = tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
if ts then
setsection(id, train, tcb[connid].ts_id, ts, {p=pos, s=connid})
end
end
end
end)
-- this time, of course, only clear the backside (cp connid)
advtrains.tnc_register_on_leave(function(pos, id, train, index)
local tcb = ildb.get_tcb(pos)
if tcb and train.path_cp[index] then
local connid = train.path_cp[index]
local ts = tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
if ts then
freesection(id, train, tcb[connid].ts_id, ts)
end
end
end)
-- those callbacks are needed to account for created and removed trains (also regarding coupling)
advtrains.te_register_on_create(function(id, train)
-- let's see what track sections we find here
local index = atround(train.index)
local pos = advtrains.path_get(train, index)
local ts_id, origin = ildb.get_ts_at_pos(pos)
if ts_id then
local ts = ildb.get_ts(ts_id)
if ts then
setsection(id, train, ts_id, ts, origin)
else
atwarn("ILDB corruption: TCB",origin," has invalid TS reference")
end
-- Make train a shunt move
train.is_shunt = true
elseif ts_id==nil then
atwarn("Train",id,": Unable to determine whether to block a track section!")
else
--atdebug("Train",id,": Outside of interlocked area!")
end
end)
advtrains.te_register_on_remove(function(id, train)
if train.il_sections then
for idx, item in ipairs(train.il_sections) do
local ts = item.ts_id and ildb.get_ts(item.ts_id)
if ts and ts.trains then
itremove(ts.trains, id)
end
end
train.il_sections = nil
end
end)
|