-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathunique.lua
632 lines (538 loc) · 14.5 KB
/
unique.lua
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
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
local parser = require'pegparser.parser'
local first = require'pegparser.first'
local disjoint = first.disjoint
local calck = first.calck
local calcfirst = first.calcfirst
local union = first.union
local getName = first.getName
local isLastAlternativeAux
local changeUnique = false
local fst, flw
local function lastSymCon (p)
if p.tag == 'con' then
return p.p2
else
return p
end
end
local function matchUnique (g, p)
if p.tag == 'char' then
return g.unique[p.p1]
elseif p.tag == 'var' and parser.isLexRule(p.p1) then
return g.unique[p.p1]
elseif p.unique and not parser.matchEmpty(p) then
return true
elseif p.tag == 'con' then
return matchUnique(g, p.p1) or matchUnique(g, p.p2)
elseif p.tag == 'ord' then
return matchUnique(g, p.p1) and matchUnique(g, p.p2)
elseif p.tag == 'plus' then
return matchUnique(g, p.p1)
else
return false
end
end
local function matchUPath (p)
if p.tag == 'char' then
return p.unique
elseif p.tag == 'var' then
return p.unique and not parser.matchEmpty(p)
elseif p.tag == 'con' then
return matchUPath(p.p1) or matchUPath(p.p2)
elseif p.tag == 'ord' then
return p.unique
elseif p.tag == 'plus' then
return p.unique
else
return false
end
end
local function matchUniqueEq (p)
if (p.tag == 'char' or p.tag == 'var') and not parser.matchEmpty(p) then
return p.uniqueEq
elseif p.tag == 'con' then
return matchUniqueEq(p.p2)
elseif p.tag == 'ord' then
return matchUniqueEq(p.p1) and matchUniqueEq(p.p2)
elseif p.tag == 'plus' then
return matchUniqueEq(p.p1)
else
return false
end
end
local function updateCountTk (v, t)
if not t[v] then
t[v] = 1
else
t[v] = t[v] + 1
end
end
local function countTk (p, t)
if p.tag == 'char' then
updateCountTk(p.p1, t)
elseif p.tag == 'set' then
for i, v in pairs(p.l) do
updateCountTk(v, t)
end
elseif p.tag == 'var' and parser.isLexRule(p.p1) then
updateCountTk(p.p1, t)
elseif p.tag == 'con' or p.tag == 'ord' then
countTk(p.p1, t)
countTk(p.p2, t)
elseif p.tag == 'star' or p.tag == 'opt' or p.tag == 'plus' then
countTk(p.p1, t)
elseif p.tag == 'and' or p.tag == 'not' then
--does not count tokens inside a predicate
return
end
end
local function printUnique (t)
local l = {}
for k, v in pairs(t) do
table.insert(l, k)
end
table.sort(l)
io.write("Unique tokens (# " .. #l .. "): ")
io.write(table.concat(l, ', '))
io.write('\n')
end
local function setUPath (p)
print("setUPath", p, p.p1, p.kind, p.upath)
if not p.upath then
changeUnique = true
end
p.upath = true
end
local function setUnique (p, v, seq)
if not v then
return
end
print("setUnique", p, p.p1, p.kind, p.unique, 'seq = ', seq)
if not p.unique then
changeUnique = true
end
p.unique = true
p.seq = seq
if v and seq then
setUPath(p)
end
end
local function uniqueTk (g)
local t = {}
for i, v in ipairs(g.plist) do
if not parser.isLexRule(v) then
countTk(g.prules[v], t)
end
end
print("Uunique")
local cont = {}
local unique = {}
for k, v in pairs(t) do
print(k, " = ", v)
unique[k] = (v == 1) or nil
if not cont[v] then
cont[v] = 1
else
cont[v] = cont[v] + 1
end
end
for i = 1, 10 do
print("Token ", i, " = ", cont[i])
end
unique['SKIP'] = nil
printUnique(unique)
return unique
end
local function addUsage(g, p)
if not g.varUsage[p.p1] then
g.varUsage[p.p1] = {}
end
table.insert(g.varUsage[p.p1], p)
end
local function countUsage(g, p)
local tag = p.tag
if tag == 'empty' or tag == 'char' or tag == 'set' or
tag == 'any' or tag == 'throw' or tag == 'def' then
return
elseif tag == 'var' and parser.isLexRule(p.p1) then
return
elseif tag == 'var' then
addUsage(g, p)
elseif tag == 'not' or tag == 'and' or tag == 'star' or
tag == 'opt' or tag == 'plus' then
countUsage(g, p.p1)
elseif tag == 'con' or tag == 'ord' then
countUsage(g, p.p1)
countUsage(g, p.p2)
else
print(p)
error("Unknown tag", p.tag)
end
end
local function varUsage (g)
g.varUsage = {}
for i, v in ipairs(g.plist) do
if not g.varUsage[v] then
g.varUsage[v] = {}
end
countUsage(g, g.prules[v])
end
end
local function printVarUsage (g)
for i, v in ipairs(g.plist) do
if not parser.isLexRule(v) then
print("Usage", v, #g.varUsage[v])
end
end
end
local function uniqueUsage (g, p)
print("testing uniqueUsage ", p.p1)
local upath, seq = true, true
for k, v in pairs(g.varUsage[p.p1]) do
print(k, v, v.unique, v.uniqueEq, v.seq)
--if not v.unique then
upath = upath and (v.unique or v.uniqueEq)
seq = seq and v.seq
end
print("Unique usage", p.p1, 'upath = ', upath, 'seq = ', seq)
return { upath = upath, seq = seq }
end
local function getSymbolsFirst (g, p)
if p.tag == 'char' or (p.tag == 'var' and parser.isLexRule(p.p1)) then
return { [p] = true }
elseif p.tag == 'var' then
return getSymbolsFirst(g, g.prules[p.p1])
elseif p.tag == 'con' then
local t = getSymbolsFirst(g, p.p1)
if parser.matchEmpty(p.p1) then
t = union(t, getSymbolsFirst(g, p.p2))
end
return t
elseif p.tag == 'ord' then
return union(getSymbolsFirst(g, p.p1), getSymbolsFirst(g, p.p2))
elseif p.tag == 'star' or p.tag == 'opt' or p.tag == 'plus' then
return getSymbolsFirst(g, p.p1)
else
return {}
end
end
local function notDisjointPref (g, p, s)
local inter = {}
local sub = {}
local pref = g.symPref[getName(p)][p]
for k, v in pairs(g.symPref[s]) do
if k ~= p then
if not disjoint(pref, v) then
inter[k] = true
end
if first.issubset(v, pref) then
sub[k] = true
end
end
end
return inter, sub
end
local function notDisjointPrefSyn (g, p)
local inter, sub = notDisjointPref(g, p, getName(p))
local s = getName(p)
for k, v in pairs(g.notDisjointFirst[s]) do
if v == 'token' then
k = '__' .. k
end
local interk, subk = notDisjointPref(g, p, k)
inter = union(inter, interk)
sub = union(sub, subk)
end
local symfirst = getSymbolsFirst(g, g.prules[p.p1])
for k, v in pairs(symfirst) do
inter[k] = nil
sub[k] = nil
end
return inter, sub
end
local function isDisjointLast (g, p, s)
local pref = g.symPref[getName(p)][p]
local inter = {}
for k, v in pairs(g.symPref[s]) do
if k ~= p then
if not disjoint(pref, v) then
inter[k] = true
print("inter = true", s, p)
end
end
end
print("isDisjointLast", p, s, next(inter))
return isLastAlternative(g, p, inter)
end
local function isLastAlternativeAux (p, last, set)
if p.tag == 'char' or p.tag == 'var' then
if set[p] then
return p
else
return last
end
elseif p.tag == 'ord' then
last = isLastAlternativeAux(p.p1, last, set)
return isLastAlternativeAux(p.p2, last, set)
elseif p.tag == 'con' then
last = isLastAlternativeAux(p.p1, last, set)
return isLastAlternativeAux(p.p2, last, set)
elseif p.tag == 'star' or p.tag == 'plus' or p.tag == 'opt' then
return isLastAlternativeAux(p.p1, last, set)
else
return last
end
end
function isLastAlternative (g, p, t)
--print("lastAlt", p, p.p1, g.symRule[p])
for k, v in pairs(t) do
if k ~= p and g.symRule[p] ~= g.symRule[k] then
--print(g.symRule[p], g.symRule[k])
return false
end
end
local aux = t[p]
t[p] = true
local res = isLastAlternativeAux(g.prules[g.symRule[p]], nil, t)
t[p] = aux
--print("lastAlt2", p.p1, res, res == p)
return res == p
end
local function isPrefixUniqueEq (g, p, inter, sub, seq)
--[==[local flw = g.symFlw[s][p]
local flwEq = {}
local nFlwEq = 0
for k, _ in pairs(prefEq) do
if not first.issubset(g.symFlw[s][k], flw) then
return false
end
end]==]
--[==[if p.previousEq then
local namePrev = getName(p.previousEq)
print("previousEq: ", p.p1, ", rule: ", g.symRule[p], ", prev: ", namePrev, p, p.previousEq)
for k, v in pairs(g.symPref[namePrev][p.previousEq]) do
io.write(k .. ' ; ')
end
io.write('\n')
end]==]
for k, _ in pairs(inter) do
if not sub[k] then
-- test if 'k' is also preceded by 'previousEq'
print("Nao foi sub", k, k.p1, g.symRule[k])
local t = { [k] = true }
if not isLastAlternative(g, p, t) then
return false
end
end
end
print("foi uniqueEq", p.p1, g.symRule[p], seq)
for k, v in pairs(g.symPref[getName(p)][p]) do
io.write(k .. ' ; ')
end
io.write('\n')
if not seq then
return
end
p.uniqueEq = true
-- TODO: review the following line, it crashes the C parser (annotates rule assignment_operator)
--p.seq = seq
if p.tag == 'var' then
print("Mais um: prefixUniqueEq uniqueUsage", p.p1, p)
g.uniqueVar[p.p1] = uniqueUsage(g, p)
end
end
-- returns a two sets: inter and sub.
-- inter has the symbols whose prefixes have some elements in commom with p
-- and some elements different
-- sub has the symbols whose prefixes are a subset of p's prefix
local function getPrefInterSub (g, p)
if p.tag == 'char' or (p.tag == 'var' and parser.isLexRule(p.p1)) then
return notDisjointPref(g, p, getName(p))
else
return notDisjointPrefSyn(g, p)
end
end
local function isPrefixUnique (g, p, seq)
local inter, sub = getPrefInterSub(g, p)
print("isPrefixUnique", next(inter), next(sub))
-- prefix is unique
local unique = next(inter) == nil
-- prefix is not unique, but appears last
if not unique then
unique = isLastAlternative(g, p, inter)
end
return unique, isPrefixUniqueEq(g, p, inter, sub, seq)
end
local function isPrefixUniqueFlw (g, p)
local inter, sub = getPrefInterSub(g, p)
local s = getName(p)
local flwInt = {}
--print("isPrefixUniqueFlw s = ", s, g.symRule[p])
for k, _ in pairs(inter) do
if not disjoint(g.symFlw[s][p], g.symFlw[getName(k)][k]) then
flwInt[k] = true
--print("colide flw", k.p1, k, g.symRule[k])
end
end
if next(flwInt) ~= nil then
--print("teve colisao")
else
return true
end
return isLastAlternative(g, p, flwInt)
end
local function uniquePrefixAux (g, p, seq)
if p.tag == 'char' or p.tag == 'var' then
--assert(not p.unique or (p.unique == true and isPrefixUnique(g, p) == true))
print("uniquePrefixAux", p.p1, p, isPrefixUnique(g, p))
setUnique(p, p.unique or isPrefixUnique(g, p, seq))
elseif p.tag == 'con' then
uniquePrefixAux(g, p.p1, seq)
if not p.p2.unique then
local last = lastSymCon(p.p1)
if (last.tag == 'char' or last.tag == 'var') and isPrefixUniqueFlw(g, last) then
setUnique(p.p2, true)
end
end
uniquePrefixAux(g, p.p2, seq or not parser.matchEmpty(p.p1))
elseif p.tag == 'ord' then
uniquePrefixAux(g, p.p1, false)
uniquePrefixAux(g, p.p2, false)
elseif p.tag == 'star' or p.tag == 'plus' or p.tag == 'opt' then
--uniquePrefixAux(g, p.p1, pflw, true)
uniquePrefixAux(g, p.p1, false)
end
end
local function uniquePrefix (g)
for i, v in ipairs(g.plist) do
if not parser.isLexRule(v) then
uniquePrefixAux(g, g.prules[v], false)
end
end
end
local function setUniqueTk (g, p)
if p.tag == 'char' or (p.tag == 'var' and parser.isLexRule(p.p1)) then
setUnique(p, g.uniqueTk[p.p1])
elseif p.tag == 'ord' or p.tag == 'con' then
setUniqueTk(g, p.p1)
setUniqueTk(g, p.p2)
elseif p.tag == 'star' or p.tag == 'plus' or p.tag == 'opt' then
setUniqueTk(g, p.p1)
end
end
local function uniquePath (g, p, uPath, flw, seq)
--print("uniquePath", p, p.tag, p.p1, uPath, p.p1.unique)
if p.tag == 'char' then
setUPath(p, uPath and seq)
elseif p.tag == 'var' and parser.isLexRule(p.p1) then
setUPath(p, uPath and seq)
elseif p.tag == 'var' and uPath then
print("unique var ", p.p1, seq, p)
setUnique(p, true, seq)
--if seq then
print("uniquePath uniqueUsage", p, p.p1)
g.uniqueVar[p.p1] = uniqueUsage(g, p)
--end
elseif p.tag == 'var' then
--print("p.p1", p.p1, #g.varUsage[p.p1])
--if matchUnique(g, g.prules[p.p1]) and #g.varUsage[p.p1] == 1 then
if matchUPath(g.prules[p.p1]) and #g.varUsage[p.p1] == 1 then
--if matchUnique(g, g.prules[p.p1]) then
print("unique var2 ", p.p1)
setUnique(p, true, seq)
end
elseif p.tag == 'con' then
uniquePath(g, p.p1, uPath, calck(g, p.p2, flw), seq)
uPath = uPath or p.p1.unique
local p1 = lastSymCon(p.p1)
if not uPath then
if p1.tag == 'var' and not parser.isLexRule(p1.p1) and matchUPath(g.prules[p1.p1]) and isDisjointLast(g, p1, getName(p1)) then
print("Vai ser agora", p1.p1, g.symRule[p1])
end
-- it seems this condition has verty little impact
if p1.tag == 'var' and not parser.isLexRule(p1.p1) and matchUPath(g.prules[p1.p1]) and isDisjointLast(g, p1, getName(p1)) then
setUnique(p1, true, seq)
uPath = true
elseif (p1.tag == 'char' or (p1.tag == 'var' and parser.isLexRule(p1.p1))) and isDisjointLast(g, p1, getName(p1)) then
setUnique(p1, true, seq)
uPath = true
end
end
if p1.uniqueEq then
print("upathEq", p1.p1)
p.p2.previousEq = p1
end
print("p.p2", uPath, p.p2.p1)
uniquePath(g, p.p2, uPath, flw, seq or not parser.matchEmpty(p.p1))
setUnique(p, uPath or p.p2.unique)
elseif p.tag == 'ord' then
local flagDisjoint = disjoint(calcfirst(g, p.p1), calck(g, p.p2, flw))
print('ord', flagDisjoint, uPath)
uniquePath(g, p.p1, flagDisjoint and uPath, flw, false)
print('p2 ord', uPath)
uniquePath(g, p.p2, uPath, flw, seq and p.p2.tag ~= 'ord')
setUnique(p, uPath or (p.p1.unique and p.p2.unique))
elseif p.tag == 'star' or p.tag == 'opt' or p.tag == 'plus' then
local flagDisjoint = disjoint(calcfirst(g, p.p1), flw)
if p.tag == 'star' or p.tag == 'plus' then
flw = union(calcfirst(g, p.p1), flw)
end
uniquePath(g, p.p1, flagDisjoint and uPath, flw, false)
if p.tag == 'plus' then
setUnique(p, uPath or p.p1.unique)
else
setUnique(p, uPath)
end
end
end
local function calcUniquePath (g)
g.uniqueVar = {}
for i, v in ipairs(g.plist) do
g.uniqueVar[v] = {}
end
fst = first.calcFst(g)
flw = first.calcFlw(g)
g.notDisjointFirst = first.notDisjointFirst(g)
g.uniqueTk = uniqueTk(g)
for i, v in ipairs(g.plist) do
if not parser.isLexRule(v) then
setUniqueTk(g, g.prules[v])
end
end
g.uniqueVar[g.plist[1]] = { upath = true, seq = true }
varUsage(g)
first.calcTail(g)
first.calcPrefix(g)
first.calcLocalFollow(g)
changeUnique = true
uniquePrefix(g)
while changeUnique do
changeUnique = false
for i, v in ipairs(g.plist) do
if not parser.isLexRule(v) then
uniquePath(g, g.prules[v], g.uniqueVar[v].upath, flw[v], g.uniqueVar[v].seq)
end
end
end
io.write("Unique vars: ")
for i, v in ipairs(g.plist) do
if g.uniqueVar[v].upath then
io.write(v .. '(' .. tostring(g.uniqueVar[v].upath) .. ';' .. tostring(g.uniqueVar[v].seq) .. ')' .. ', ')
end
end
io.write('\n')
io.write("matchUPath: ")
for i, v in ipairs(g.plist) do
if matchUPath(g.prules[v]) then
io.write(v .. ', ')
end
end
io.write('\n')
end
return {
uniqueTk = uniqueTk,
calcUniquePath = calcUniquePath,
matchUnique = matchUnique,
matchUPath = matchUPath,
matchUniqueEq = matchUniqueEq,
}