summaryrefslogtreecommitdiff
path: root/wiki/Chat_log/20150804-io-chatlog
blob: 39f02228ada4aaabf03718730acd922e0362e6e1 (plain)
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
--- Log opened Tue Aug 04 10:42:04 2015
10:42 -!- wsa_ [~wsa@p4FE2576E.dip0.t-ipconnect.de] has joined #periperi
10:42 -!- Irssi: #periperi: Total of 4 nicks [0 ops, 0 halfops, 0 voices, 4 normal]
10:42 -!- Irssi: Join to #periperi was synced in 1 secs
10:43 < wsa_> konnichi wa
10:45 < morimoto> konnichi wa
10:46 < wsa_> hiya morimoto-san
10:46 < wsa_> how is copy/pasting windows code? ;)
10:47 < morimoto> Hehe :) it works very well for me (at this point)
10:48 -!- shimoda [~shimoda@relprex1.renesas.com] has joined #periperi
10:48 < wsa_> hello shimoda-san
10:48 < shimoda> hello wolfram-san!
10:49 -!- geertu [~geert@188.188.25.36] has joined #periperi
10:50 < geertu> Woops, I lost all networking (Internet/TV) at home ca, 30 minutes ago.
10:51 < geertu> Back online through 3G and Wifi-hotspot, but 3G coverage inside the house is really bad (and too much rain outside)
10:51 < wsa_> hi geert
10:52 < wsa_> i'll send some of the sunshine here towards you
10:52 < geertu> thx!
10:52 < geertu> Don't forget to keep some sunshine for your holidays next week ;-)
10:52 < wsa_> (and the plants could need a bit of rain, too)
10:52 < wsa_> :)
10:53 < wsa_> luckily, i am not so weather-dependent
10:53 < wsa_> the area is nice in any case
10:56 < morimoto> In these days, Japan is super very hot days. you can bring it
10:57 < geertu> yesterday it was super very hot in Belgium too.
10:57 < geertu> Today it's 10 degrees colder, and rainy
10:57 < wsa_> 33 degrees here, today
10:58 < morimoto> not so different ? Japan-Tokyo is 34 - 35 degrees
10:59 < wsa_> but probably a lot more humid
10:59 < morimoto> YES !!
10:59 < wsa_> :)))
10:59 < geertu> yeah... but not sufficiently humid to enjoy it (-> swimming pool)
11:00 -!- horms [~horms@121-80-222-183f1.hyg1.eonet.ne.jp] has joined #periperi
11:00 < wsa_> hello simon
11:00 < horms> wsa_: are we having an I/O group meeting about now?
11:00 < wsa_> yes, you got perfect timing
11:00 -!- geertu_ [~geert@d54C15D57.access.telenet.be] has joined #periperi
11:01 < horms> excellent
11:01 < wsa_> geert has lagging problems?
11:02 < geertu> Yeah, dah Internet is back
11:02  * geertu switches back to broadband
11:02 -!- geertu [~geert@188.188.25.36] has quit Quit: leaving
11:02 < wsa_> \o/
11:02 < geertu_> hi
11:02 -!- geertu_ is now known as geertu
11:03 < wsa_> so, i guess we can start now
11:03 < geertu> Doh, and the 3G just managed to pull in drm/du/vsp1-kms...
11:03 < wsa_> thanks for coming to this very first IO Group IRC Chat Meeting
11:03 < wsa_> let's make history!
11:03 < wsa_> ahem
11:03 < wsa_> :)
11:04 < wsa_> the topics and order i'd propose are:
11:04 < wsa_> 1) Needed steps for Gen3 (should be easily dealt with)
11:05 < wsa_> 2) status of SCIF (DMA...)
11:05 < wsa_> 3) discuss the current list of tasks
11:05 < wsa_> the last one will probably evolve into questions like "tracking the BSP" and "reproducing issues"
11:06 < geertu> git log
11:06  * geertu sorry wrong window
11:06 < wsa_> any comments to that? any additions?
11:06 < horms> I have one more, small item: scif patches in Rcar-Gen2 BSP 1.9.5
11:07 < horms> if we have time
11:07 < geertu> Doesn't that fall under 2)?
11:07 < horms> probably :)
11:07 < wsa_> I'd think so, too
11:07 < horms> ok, that is fine by me
11:08 < wsa_> so, let's start with 1)
11:09 < wsa_> by mail, shimoda-san said that I2C enablement patches for Gen3 are needed, prototypes will do
11:09 < wsa_> I can do this before I go on vacation
11:09 < morimoto> very good !
11:09 < wsa_> the cores have additions, but should be (tm) backwards compatible
11:09 < horms> ok, iirc that was around the 11th
11:10 < wsa_> it will be a matter of adding compatibles, I can do this today
11:10 < horms> ok, sounds good from my pv
11:11 < horms> ok, sounds good from my pov
11:11 < wsa_> anything else for Gen3?
11:11 < wsa_> some news since yesterday? :)
11:11 < shimoda> about etheravb
11:11 < shimoda> it needs until 25th august
11:12 < shimoda> yesterday?
11:13 < horms> is driver work required for etheravb, over what is queued up for v4.3 in net-next
11:13 < horms> ?
11:13 < wsa_> shimoda: sorry, not yesterday, but since your mail
11:14 < morimoto> horms: do you mean for Gen3 BSP ?
11:14 < horms> yes
11:14 < wsa_> horms: have you checked the Gen3 docs if EtherAVB is the same as Gen2?
11:15 < horms> wsa_: ah, not as much as i should have
11:15 < horms> is that an ai for me?
11:15 < wsa_> can you do this please?
11:15 < horms> sure
11:16 < morimoto> for BSP, we need EtherAVB but no one is listed inhouse Renesas.
11:16 < horms> ok, i guess it defaults to cogent
11:17 < wsa_> still, will be nice to know if the IP cores are the same or how they differ
11:17 < morimoto> I guess integration is not a big deal (if it has compatible)
11:17 < horms> wsa_: agreed
11:17 < morimoto> If it needs driver update, then Cogent ?
11:18 < horms> morimoto: I don't know. But they have handled the driver so far.
11:18 < morimoto> yes
11:18 < wsa_> hmmm, who knows this answer?
11:18 < horms> If we have to rely on Cogent for driver updates for the Gen3 BSP then I think we have a problem
11:19 < horms> wsa_: probably Munakata-san, but my feeling is it would be best to talk to Magnus.
11:19 < wsa_> okay
11:19 < horms> in any case, i don't think we can resolve it here
11:20 < wsa_> yes
11:20 < wsa_> I'd think, let's check the IP cores, find out responsibilities and let's see from there...
11:20 < shimoda> at the moment, bsp team is debugging etheravb
11:20 < horms> good plan
11:21 < horms> shimoda: on gen2 hw?
11:21 < wsa_> shimoda: gen2 or gen3?
11:21 < wsa_> :D
11:21 < shimoda> sorry, on gen3
11:22 < wsa_> so, it might even be a moving target
11:22 < shimoda> according to his email, gen3 needs some additional setting in avb and phy
11:23 < shimoda> so, for upstreaming, we need his help
11:23 < horms> komatsu-san?
11:24 < shimoda> His name is Nagai-san, sometimes he sent email to net ML
11:24 < horms> oh ok
11:25 < horms> should I/we talk to him?
11:25 < horms> He may be aware of some issues that are outside of what is described in the documentation we have
11:26 < shimoda> yes, i\I will talk to him about gen3 avb
11:26 < horms> excellent, thanks
11:26 < wsa_> thank you
11:26 < wsa_> i will talk to magnus about the responsibility
11:27 < wsa_> and simon will have a look at the docs
11:27 < wsa_> so, last call for Gen3 topics
11:28 < morimoto> 1 question about I2C
11:28 < morimoto> I guess Gen3 has 2 type of I2C ?
11:28 < wsa_> yes, 7x I2C and 1x IIC
11:28 < morimoto> which I2C do you enable before your vacation ?
11:29 < wsa_> both
11:29 < morimoto> Cool !! Thanks
11:29 < wsa_> it is just compatibles
11:29 < wsa_> :)
11:29 < shimoda> for gen3, we need some code of "SPI" until 1st Sept.
11:30 < shimoda> maybe MSIOF
11:30 < geertu> The real MSIOF or the "slave-only" MSIOF?
11:30 < shimoda> it is the real MSIOF
11:32 < geertu> (from "Gen3 work") MSIOF: Looks identical to Gen2
11:32 < geertu> (from "Gen3 work" email) MSIOF: Looks identical to Gen2
11:32 < wsa_> okay, so another compatible update
11:32 < wsa_> geert: can you do this till Sep, 1st? ;)
11:32 < geertu> RPC (2x QSPI) needs a new driver
11:32 < geertu> Sure, but currently I cannot test it
11:33 < wsa_> i know
11:33 < wsa_> let's see how the hardware status is 3 weeks
11:34 < wsa_> if there is no HW, then we do the prototypes again
11:34 < wsa_> OK, I'd like to move on to the next topic
11:34 < horms> i would guess it wiil be similar to now :^)
11:35 < wsa_> i wouldn't bet much against it ;)
11:35 < geertu> I guess they will produce hardware during the Renesas Summer Holiday Week?
11:35  * geertu wonders if that needs a smiley or not...
11:35 < morimoto> In our plan we will have 1 H3 board tomorrow
11:35 < wsa_> so, SCIF
11:36 < geertu> Great!
11:36 < morimoto> and Magnus will setup it on his remote access machine
11:36 < wsa_> WOW!
11:36 < geertu> Great! (x2)
11:36 < morimoto> (If we can have board)
11:36 < shimoda> sorry, about RPC, I'm not sure but BSP team doesn't support it because secure
11:36 < geertu> Just before I left for holidays, I posted a big patch series for SCIF DMA
11:37 < wsa_> (so, I will wait a few days before posting the I2C Gen3 patches, maybe i can test after all)
11:37 < geertu> It's sort of working now
11:37 < geertu> But it needs more rework and cleanup
11:37 < horms> excellent
11:37 < wsa_> great
11:37 < morimoto> cool
11:37 < horms> I believe the state of scif in the gen2 BSP is a point of concern
11:38 < wsa_> does the rework need DMA framework changes?
11:38 < geertu> I think half of the patches in that series can be sent out for integration, but I didn't have time to do the split when I sent the whole series.
11:38 < geertu> Good question!
11:39 < geertu> It relies on "modern" residu handling, which the shdmac driver doesn't provide (do we care? I posted an RFC patch)
11:40 < geertu> And I had to change the SCIF driver to not resubmit DMA descriptors, as rcar-dmac doesn't support that
11:40 < geertu> Still waiting for Laurent's response to my rcar-dmac emails...
11:40 < horms> which hardware do we use shdma on?
11:41 < geertu> "sh-dma-engine" (sh, legacy shmobile)
11:41 < geertu> "renesas,shdma-r8a73a4" (r8a73a4)
11:41 < geertu> "hpb-dma-engine" (legacy r8a777x)
11:41 < geertu> "sudmac" (unused, referred to by r8a66597_udc on ecovec24/kfr2r09/se)
11:42 < geertu> That was the long answer
11:42 < horms> ok
11:42 < geertu> The short answer is: nothing that uses DT and multi-platform
11:42 < horms> so my feeling is that if we aren't using it on multiplatform gen2 (or 3) then its not so important
11:42 < geertu> "renesas,shdma-r8a73a4" is not really used
11:43 < wsa_> does the BSP use rcar-dmac meanwhile?
11:43 < horms> good question
11:43 < horms> they are using multiplatform (i believe)
11:43 < horms> and they only care about gen2
11:43 < horms> so it seems likely
11:43 < horms> but i don't know for sure
11:44 < geertu> renesas-backports/bsp/v3.10.31-ltsi/rcar-gen2-1.9.5:drivers/tty/serial/sh-sci.c still has
11:44 < horms> my feeling about the relationship between this work and the bsp is
11:44 < geertu>                 struct shdma_desc *sh_desc = container_of(desc,
11:44 < geertu>                                         struct shdma_desc, async_tx);
11:44 < horms> that they have been doing some work to try to stablalise sh-sci
11:44 < geertu> Now, you can still do the container_of() on an rcar-dmac dma_desc object, and get bogus results :-)
11:45 < horms> and i see they have some fixes in the latest v1.9.5 bsp
11:45 < geertu> So I don't think they use rcar-dmac.
11:45 < horms> now, if we can provide them a nice set of fixes to solve their problems then that would be great
11:45 < wsa_> horms: they seem to do, there is even a patch for it in the BSP (which is already upstream, too)
11:45 < horms> but if its invasive, touching all sorts of code, then its not really appropriate for the gen2 bsp
11:46 < shimoda> they use rcar-dmac for audio and some customers uses it for sdhi/mmc
11:46 < shimoda> they = BSP team
11:46 < geertu> #define r8a7791_register_sys_dmac(id)                           \
11:46 < geertu>         platform_device_register_resndata(                      \
11:46 < geertu>                 &platform_bus, "sh-dma-engine", 2 + id,         \
11:46 < geertu>                 &r8a7791_sys_dmac_resources[id * 3],    id * 1 + 3,     \
11:46 < geertu>                 &r8a7791_sys_dmac_platform_data,                \
11:46 < geertu>                 sizeof(r8a7791_sys_dmac_platform_data))
11:47 < geertu> So they use shdmac with hardcoded chabnnels in board code
11:47 < horms> with multiplatform boots?
11:47 < wsa_> so, if they use rcar-dmac already, it would be a technically wise decision if they would use rcar-dmac for serial, too, no?
11:48 < horms> it might be wise
11:48 < shimoda> sorry, they use both rcar-dmac and shdmac
11:48 < horms> but feature-wise the tree is frozen
11:48 < geertu> I once asked Kaneko-san explicitly with which dmac driver his sh-sci patches were tested, as IMHO the patch couldn't work with rcar-dmac (don't remember why, though)
11:48 < horms> geertu: just to clarify, those would be BSP-team patches, he isn't on the team
11:49 < wsa_> horms: is switching to rcar-dmac a feature or a bugfix?
11:49 < horms> i guess it might be either
11:49 < wsa_> if it was a feature, then the bugfix would be to keep on hacking on shdma
11:50 < wsa_> which is not what we really want being the upstram focused team, or?
11:50 < wsa_> (I might be wrong, but I recall Laurent not being very happy with shdma)
11:50 < horms> well, i think that this is one area where there has been some divergence between upstream and the BSP
11:50 < horms> and in my opinion that is not ideal
11:51 < horms> but i wonder what the best way forwards is
11:51 < geertu> r8a7791.dtsi has no "dmas" properties, except in the rcar_sound node
11:51 < horms> i beliwve we have trouble in both mainline and bsp
11:51 < geertu> yes
11:51 < horms> and obviously we wish to fix the former properly
11:51 < geertu> sh-sci needs more care.
11:52 < horms> and ideally take that to the bsp, but i think the devil is in the details
11:52 < horms> regarding how risky the bsp changes might be in terms of stability
11:52 < horms> an unfortunate side effect of this situation is that
11:52 < wsa_> if we would be able to fix it upstream by largely modifying the serial driver only, my hope would be that we could then convince the BSP team to switch over?
11:52 < geertu> the SCIF DNA code is still so complex that there are no obvious deficiencies (I prefer: so simple that there are obviously no deficiencies)
11:52 < horms> its not entirely obvious when a fix for mainline or the BSP is applicable to the other
11:53 < wsa_> horms: true
11:53 < wsa_> there might be side effects
11:53 < geertu> The BSP has several fixes for race conditions, but I don't think they fix the root cause.
11:54 < horms> that would not surprise me
11:54 < geertu> (The same is true for my last RFC series, though)
11:54 < wsa_> but if those side effects come from bugs, then fixing them obviously will change behaviour of the driver
11:54 < horms> they could be positive behavoural changes :^)
11:55 < wsa_> so far, geert's series is completely self contained to the scif driver
11:55 < geertu> Most changes don't look like real behavior changes. Just fixes for rcae conditions.
11:56 < geertu> wsa_: yes
11:56 < shimoda> by the way, I tested geert-san's scif-dma-v2 patch set last week
11:56 < shimoda> and I sent some patches to the ML :)
11:56 < geertu> Thanks for testing, and for the patches.
11:57 < geertu> Which SCI variant(s) did you test?
11:57 < shimoda> i tested on Lager (SCIFA) and koelsch (SCIF)
11:58 < geertu> Good, thanks!
11:58 < horms> ok, i guess we could look at providing some fixes to the bsp team
11:58 < wsa_> my feeling is that it would be worthwhile fixing the remaining issues until we understand what was going wrong and then "sell" our fixes to the BSP team
11:58 < horms> if so i'd like some co-ordination between your changes (and mainine in general) and what is in BSP 1.9.5
11:58 < geertu> I think it's too early for that
11:59 < horms> i guess you and i could co-ordinate that when the timing is right
11:59 < geertu> Yes, I want to fix the remaining issues first
11:59 < horms> ok, no problem regarding waiting
11:59 < geertu> Ater that I can created a "nicer" series.
11:59 < horms> could you take a look at the new bsp (i pushed it on Saturday) on the off chance any of the scif patches there are of use to you
11:59 < geertu> The current one has too many fixes here, fixes there, ... some of them may not be needed inthe end
11:59 < wsa_> that would hopefully also reduce the gap between BSP & mainline
12:00 < horms> i think you only need to look back as far as v1.9.4 as kaneko-san has already fed the older ones to the ml
12:00 < geertu> I already had a quick look, and none of them looked that exciting
12:00 < horms> thanks
12:00 < horms> that was my expectation
12:00 < horms> I will tell Kaneko-san not to bother posting them to the ml
12:02 < wsa_> ok
12:02 < wsa_> so, geert will push out a nicer series
12:02 < geertu> horms: Thanks, sometimes I'm afraid Greg would apply them ;-)
12:02 < horms> haha
12:02 < wsa_> that can happen
12:02 < wsa_> :)
12:03 < horms> as a precaution i already told him not to post them unless he hears otherwise from me :^)
12:03 < wsa_> and hopefully after the new series, we know a bit more how to deal with the issues
12:04 < horms> sounds reasonable to me
12:04 < wsa_> and then we can have a look what would be best for BSP
12:04 < horms> yes, that can come last
12:04 < horms> imho
12:05 < geertu> I hope scif work will become a bit easier now it's at least working a bit
12:05 < horms> hopefully...
12:05 < geertu> Before it was really frustrating, as any small failure somewhere would lead to one out of many possible crashes due to race conditions
12:05 < horms> ouch
12:05 < geertu> On every retry, I had a different crash ;-)
12:06 < wsa_> yes, working on old serial drivers always deserves a "braveness medal"
12:06 < geertu> I'm wondering how SCIF DMA ever worked. I guess SMP=n helped a lot...
12:07 < wsa_> ok, so much about scif?
12:07 < geertu> Does SCIF DMA work on e.g. Ecovec?
12:07 < horms> i may have one if you really want to find out
12:08 < shimoda> geert-san: I don't know
12:08 < geertu> we never wired up scif dma on e.g. armadillo and kzm9g legacy
12:09 < geertu> wsa_ scif EOF
12:09 < wsa_> :)
12:10 < wsa_> so 3) task list
12:10 < wsa_> I sent it around by mail
12:10 < wsa_> I guess you would have replied if you really missed an item there
12:10 < geertu> Sorry, one more SCIF DMA thing
12:10 < wsa_> ok
12:11 < geertu> The people from Visteon contacted me, as they're interested in it for their product.
12:11 < geertu> For a 4 Mbps link to Bluetooth on SCIFA
12:12 < geertu> Just wanted to let you know...
12:12 < wsa_> set them up for testing ;)
12:13 < wsa_> ok, other comments regarding the io group todo list?
12:13 < geertu> the list looks fine to me
12:14 < morimoto> Sorry, is this "todo list" this ? https://osdr.renesas.com/projects/linux-kernel-development/wiki/Io-todo-list
12:14 < wsa_> yes
12:16 < wsa_> so, when looking for further tasks to keep an eye on, I tried to scan through the BSP
12:16 < wsa_> I concentrated on I2C and MMC for starters
12:17 < wsa_> Conclusion: most patches would already need some investigation to simply understand the problem :)
12:18 < wsa_> and in most cases it is very difficult because there is not enough information to reproduce the issue
12:18 < wsa_> so while "scanning the BSP" sounds good in general
12:18 < wsa_> I am afraid we won't have the bandwidth to do it in general
12:19 < horms> i can get the BSP scanned if you like
12:19 < horms> but my feeling is that its already been scanned up until v1.9.4 (and v1.9.5 but not posted yet)
12:21 < horms> of course you are also welcome to look over it as much as you like
12:22 < wsa_> who scans it?
12:22 < wsa_> kaneko-san?
12:22 < horms> Kaneko-san and I
12:22 < wsa_> ok
12:23 < horms> To be honest I think we got all the fat last year
12:23 < wsa_> I wondered about some MMC related patches
12:24 < wsa_> but my suggestion was anyhow: we bring it to this table here
12:24 < horms> anything in particular
12:24 < horms> i recall some relating to timeouts
12:24 < horms> ok
12:25 < wsa_> if there is something easy, we just pick it
12:25 < horms> as i mentioned earlier the BSP is sort of frozen now
12:25 < wsa_> if there is something "interesting", we can discuss it here or by mail
12:25 < horms> so the reate of patches being added has dropped of significantly since... i guess the begninnging of the year
12:25 < wsa_> ok
12:25 < horms> so there isn't such a large crop of changes to pick from
12:25 < wsa_> "bugfix only" mode?
12:26 < horms> something like that
12:26 < horms> i'm not sure of their exact criteria
12:26 < horms> but its mostly if not exclusively bug fixes from my observations
12:27 < horms> there are a few changes waiting feedback from the BSP team
12:27 < horms> i can try chasing them up
12:27 < horms> which may yeild a few more fixes for mainline
12:27 < wsa_> do you know if there will be another "major version" somewhen or is that "the" BSP foreverß
12:27 < wsa_> ?
12:27 < horms> of course for gen3, yes
12:28 < horms> and probably that will support gen2 in some way (perhaps not officually, but it should boot the boards)
12:28 < wsa_> i understand
12:28 < horms> but for gen2 bsp, i believe the plan is no more major updates
12:28 < horms> actually, i think the plan was no more updates at all: but then some bugs turned up...
12:29 < wsa_> damn software ;)
12:29 < horms> yeah, its like that sometimes
12:31 < wsa_> okay, I think we are done with the planned topics
12:31 < wsa_> anything else which needs to be said now?
12:31 < horms> right on time!
12:32 < shimoda> I have no topic at the moment :)
12:33 < wsa_> cool
12:33 < wsa_> so, next meeting probably the week after my holiday
12:33 < wsa_> I will send out a mail
12:33 < horms> enjoy your break
12:34 < wsa_> oh, i will
12:34 < wsa_> but until then, there is another week of work left
12:34 < wsa_> ;)
12:34 < wsa_> thank you guys!
12:34 < morimoto> BTW, Renesas will have summer vacation 10th - 14th Aug
12:34 < shimoda> thank you!
12:34 < wsa_> have a great day/evening!
12:34 < wsa_> morimoto: thanks, noted
12:35 < morimoto> Please let me know if you need something. this week is good for me
12:35 < geertu> Bye!
12:35 < horms> BTW, I will have vacation 10th, 11th (but i can release topic branches as required)
12:35 < shimoda> byebye!
12:35 < morimoto> horms: ok
12:36 < morimoto> bye ! thank you
12:36 -!- morimoto [~user@relprex3.renesas.com] has left #periperi ["ERC Version 5.3 (IRC client for Emacs)"]
12:36 -!- shimoda [~shimoda@relprex1.renesas.com] has quit Quit: WeeChat 0.4.2
12:36 < horms> geertu: will you be sending v4 a little later?
12:37 < horms> if not i won't poll for it :)
12:37 < geertu> horms: yes, I was a bit delayed by the Internet link breakdown
12:37 < horms> understandable
12:38  * geertu has released renesas-drivers-2015-08-04-v4.2-rc5
12:38 < horms> anyway no particular rush here
12:38 < horms> i'll check my email a bit later
12:38 < geertu> will do v4 after lunch
12:38 < horms> good plan
12:38 < horms> i'll disappear for a bit
12:42 < wsa_> me too
--- Log closed Tue Aug 04 12:42:40 2015
id='n1119' href='#n1119'>1119 1120 1121 1122 1123
/*
 * Copyright 2004 The Unichrome Project. All Rights Reserved.
 * Copyright 2005 Thomas Hellstrom. All Rights Reserved.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sub license,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the
 * next paragraph) shall be included in all copies or substantial portions
 * of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
 * THE AUTHOR(S), AND/OR THE COPYRIGHT HOLDER(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 *
 * Author: Thomas Hellstrom 2004, 2005.
 * This code was written using docs obtained under NDA from VIA Inc.
 *
 * Don't run this code directly on an AGP buffer. Due to cache problems it will
 * be very slow.
 */

#include "via_3d_reg.h"
#include "drmP.h"
#include "drm.h"
#include "via_drm.h"
#include "via_verifier.h"
#include "via_drv.h"

typedef enum {
	state_command,
	state_header2,
	state_header1,
	state_vheader5,
	state_vheader6,
	state_error
} verifier_state_t;

typedef enum {
	no_check = 0,
	check_for_header2,
	check_for_header1,
	check_for_header2_err,
	check_for_header1_err,
	check_for_fire,
	check_z_buffer_addr0,
	check_z_buffer_addr1,
	check_z_buffer_addr_mode,
	check_destination_addr0,
	check_destination_addr1,
	check_destination_addr_mode,
	check_for_dummy,
	check_for_dd,
	check_texture_addr0,
	check_texture_addr1,
	check_texture_addr2,
	check_texture_addr3,
	check_texture_addr4,
	check_texture_addr5,
	check_texture_addr6,
	check_texture_addr7,
	check_texture_addr8,
	check_texture_addr_mode,
	check_for_vertex_count,
	check_number_texunits,
	forbidden_command
} hazard_t;

/*
 * Associates each hazard above with a possible multi-command
 * sequence. For example an address that is split over multiple
 * commands and that needs to be checked at the first command 
 * that does not include any part of the address.
 */

static drm_via_sequence_t seqs[] = {
	no_sequence,
	no_sequence,
	no_sequence,
	no_sequence,
	no_sequence,
	no_sequence,
	z_address,
	z_address,
	z_address,
	dest_address,
	dest_address,
	dest_address,
	no_sequence,
	no_sequence,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	tex_address,
	no_sequence
};

typedef struct {
	unsigned int code;
	hazard_t hz;
} hz_init_t;

static hz_init_t init_table1[] = {
	{0xf2, check_for_header2_err},
	{0xf0, check_for_header1_err},
	{0xee, check_for_fire},
	{0xcc, check_for_dummy},
	{0xdd, check_for_dd},
	{0x00, no_check},
	{0x10, check_z_buffer_addr0},
	{0x11, check_z_buffer_addr1},
	{0x12, check_z_buffer_addr_mode},
	{0x13, no_check},
	{0x14, no_check},
	{0x15, no_check},
	{0x23, no_check},
	{0x24, no_check},
	{0x33, no_check},
	{0x34, no_check},
	{0x35, no_check},
	{0x36, no_check},
	{0x37, no_check},
	{0x38, no_check},
	{0x39, no_check},
	{0x3A, no_check},
	{0x3B, no_check},
	{0x3C, no_check},
	{0x3D, no_check},
	{0x3E, no_check},
	{0x40, check_destination_addr0},
	{0x41, check_destination_addr1},
	{0x42, check_destination_addr_mode},
	{0x43, no_check},
	{0x44, no_check},
	{0x50, no_check},
	{0x51, no_check},
	{0x52, no_check},
	{0x53, no_check},
	{0x54, no_check},
	{0x55, no_check},
	{0x56, no_check},
	{0x57, no_check},
	{0x58, no_check},
	{0x70, no_check},
	{0x71, no_check},
	{0x78, no_check},
	{0x79, no_check},
	{0x7A, no_check},
	{0x7B, no_check},
	{0x7C, no_check},
	{0x7D, check_for_vertex_count}
};

static hz_init_t init_table2[] = {
	{0xf2, check_for_header2_err},
	{0xf0, check_for_header1_err},
	{0xee, check_for_fire},
	{0xcc, check_for_dummy},
	{0x00, check_texture_addr0},
	{0x01, check_texture_addr0},
	{0x02, check_texture_addr0},
	{0x03, check_texture_addr0},
	{0x04, check_texture_addr0},
	{0x05, check_texture_addr0},
	{0x06, check_texture_addr0},
	{0x07, check_texture_addr0},
	{0x08, check_texture_addr0},
	{0x09, check_texture_addr0},
	{0x20, check_texture_addr1},
	{0x21, check_texture_addr1},
	{0x22, check_texture_addr1},
	{0x23, check_texture_addr4},
	{0x2B, check_texture_addr3},
	{0x2C, check_texture_addr3},
	{0x2D, check_texture_addr3},
	{0x2E, check_texture_addr3},
	{0x2F, check_texture_addr3},
	{0x30, check_texture_addr3},
	{0x31, check_texture_addr3},
	{0x32, check_texture_addr3},
	{0x33, check_texture_addr3},
	{0x34, check_texture_addr3},
	{0x4B, check_texture_addr5},
	{0x4C, check_texture_addr6},
	{0x51, check_texture_addr7},
	{0x52, check_texture_addr8},
	{0x77, check_texture_addr2},
	{0x78, no_check},
	{0x79, no_check},
	{0x7A, no_check},
	{0x7B, check_texture_addr_mode},
	{0x7C, no_check},
	{0x7D, no_check},
	{0x7E, no_check},
	{0x7F, no_check},
	{0x80, no_check},
	{0x81, no_check},
	{0x82, no_check},
	{0x83, no_check},
	{0x85, no_check},
	{0x86, no_check},
	{0x87, no_check},
	{0x88, no_check},
	{0x89, no_check},
	{0x8A, no_check},
	{0x90, no_check},
	{0x91, no_check},
	{0x92, no_check},
	{0x93, no_check}
};

static hz_init_t init_table3[] = {
	{0xf2, check_for_header2_err},
	{0xf0, check_for_header1_err},
	{0xcc, check_for_dummy},
	{0x00, check_number_texunits}
};

static hazard_t table1[256];
static hazard_t table2[256];
static hazard_t table3[256];

static __inline__ int
eat_words(const uint32_t ** buf, const uint32_t * buf_end, unsigned num_words)
{
	if ((buf_end - *buf) >= num_words) {
		*buf += num_words;
		return 0;
	}
	DRM_ERROR("Illegal termination of DMA command buffer\n");
	return 1;
}

/*
 * Partially stolen from drm_memory.h
 */

static __inline__ drm_local_map_t *via_drm_lookup_agp_map(drm_via_state_t * seq,
							  unsigned long offset,
							  unsigned long size,
							  drm_device_t * dev)
{
#ifdef __linux__
	struct list_head *list;
	drm_map_list_t *r_list;
#endif
	drm_local_map_t *map = seq->map_cache;

	if (map && map->offset <= offset
	    && (offset + size) <= (map->offset + map->size)) {
		return map;
	}
#ifdef __linux__
	list_for_each(list, &dev->maplist->head) {
		r_list = (drm_map_list_t *) list;
		map = r_list->map;
		if (!map)
			continue;
#else
	TAILQ_FOREACH(map, &dev->maplist, link) {
#endif
		if (map->offset <= offset
		    && (offset + size) <= (map->offset + map->size)
		    && !(map->flags & _DRM_RESTRICTED)
		    && (map->type == _DRM_AGP)) {
			seq->map_cache = map;
			return map;
		}
	}
	return NULL;
}

/*
 * Require that all AGP texture levels reside in the same AGP map which should 
 * be mappable by the client. This is not a big restriction.
 * FIXME: To actually enforce this security policy strictly, drm_rmmap 
 * would have to wait for dma quiescent before removing an AGP map. 
 * The via_drm_lookup_agp_map call in reality seems to take
 * very little CPU time.
 */

static __inline__ int finish_current_sequence(drm_via_state_t * cur_seq)
{
	switch (cur_seq->unfinished) {
	case z_address:
		DRM_DEBUG("Z Buffer start address is 0x%x\n", cur_seq->z_addr);
		break;
	case dest_address:
		DRM_DEBUG("Destination start address is 0x%x\n",
			  cur_seq->d_addr);
		break;
	case tex_address:
		if (cur_seq->agp_texture) {
			unsigned start =
			    cur_seq->tex_level_lo[cur_seq->texture];
			unsigned end = cur_seq->tex_level_hi[cur_seq->texture];
			unsigned long lo = ~0, hi = 0, tmp;
			uint32_t *addr, *pitch, *height, tex;
			unsigned i;
			int npot;

			if (end > 9)
				end = 9;
			if (start > 9)
				start = 9;

			addr =
			    &(cur_seq->t_addr[tex = cur_seq->texture][start]);
			pitch = &(cur_seq->pitch[tex][start]);
			height = &(cur_seq->height[tex][start]);
			npot = cur_seq->tex_npot[tex];
			for (i = start; i <= end; ++i) {
				tmp = *addr++;
				if (tmp < lo)
					lo = tmp;
				if (i == 0 && npot)
					tmp += (*height++ * *pitch++);
				else
					tmp += (*height++ << *pitch++);
				if (tmp > hi)
					hi = tmp;
			}

			if (!via_drm_lookup_agp_map
			    (cur_seq, lo, hi - lo, cur_seq->dev)) {
				DRM_ERROR
				    ("AGP texture is not in allowed map\n");
				return 2;
			}
		}
		break;
	default:
		break;
	}
	cur_seq->unfinished = no_sequence;
	return 0;
}

static __inline__ int
investigate_hazard(uint32_t cmd, hazard_t hz, drm_via_state_t * cur_seq)
{
	register uint32_t tmp, *tmp_addr;

	if (cur_seq->unfinished && (cur_seq->unfinished != seqs[hz])) {
		int ret;
		if ((ret = finish_current_sequence(cur_seq)))
			return ret;
	}

	switch (hz) {
	case check_for_header2:
		if (cmd == HALCYON_HEADER2)
			return 1;
		return 0;
	case check_for_header1:
		if ((cmd & HALCYON_HEADER1MASK) == HALCYON_HEADER1)
			return 1;
		return 0;
	case check_for_header2_err:
		if (cmd == HALCYON_HEADER2)
			return 1;
		DRM_ERROR("Illegal DMA HALCYON_HEADER2 command\n");
		break;
	case check_for_header1_err:
		if ((cmd & HALCYON_HEADER1MASK) == HALCYON_HEADER1)
			return 1;
		DRM_ERROR("Illegal DMA HALCYON_HEADER1 command\n");
		break;
	case check_for_fire:
		if ((cmd & HALCYON_FIREMASK) == HALCYON_FIRECMD)
			return 1;
		DRM_ERROR("Illegal DMA HALCYON_FIRECMD command\n");
		break;
	case check_for_dummy:
		if (HC_DUMMY == cmd)
			return 0;
		DRM_ERROR("Illegal DMA HC_DUMMY command\n");
		break;
	case check_for_dd:
		if (0xdddddddd == cmd)
			return 0;
		DRM_ERROR("Illegal DMA 0xdddddddd command\n");
		break;
	case check_z_buffer_addr0:
		cur_seq->unfinished = z_address;
		cur_seq->z_addr = (cur_seq->z_addr & 0xFF000000) |
		    (cmd & 0x00FFFFFF);
		return 0;
	case check_z_buffer_addr1:
		cur_seq->unfinished = z_address;
		cur_seq->z_addr = (cur_seq->z_addr & 0x00FFFFFF) |
		    ((cmd & 0xFF) << 24);
		return 0;
	case check_z_buffer_addr_mode:
		cur_seq->unfinished = z_address;
		if ((cmd & 0x0000C000) == 0)
			return 0;
		DRM_ERROR("Attempt to place Z buffer in system memory\n");
		return 2;
	case check_destination_addr0:
		cur_seq->unfinished = dest_address;
		cur_seq->d_addr = (cur_seq->d_addr & 0xFF000000) |
		    (cmd & 0x00FFFFFF);
		return 0;
	case check_destination_addr1:
		cur_seq->unfinished = dest_address;
		cur_seq->d_addr = (cur_seq->d_addr & 0x00FFFFFF) |
		    ((cmd & 0xFF) << 24);
		return 0;
	case check_destination_addr_mode:
		cur_seq->unfinished = dest_address;
		if ((cmd & 0x0000C000) == 0)
			return 0;
		DRM_ERROR
		    ("Attempt to place 3D drawing buffer in system memory\n");
		return 2;
	case check_texture_addr0:
		cur_seq->unfinished = tex_address;
		tmp = (cmd >> 24);
		tmp_addr = &cur_seq->t_addr[cur_seq->texture][tmp];
		*tmp_addr = (*tmp_addr & 0xFF000000) | (cmd & 0x00FFFFFF);
		return 0;
	case check_texture_addr1:
		cur_seq->unfinished = tex_address;
		tmp = ((cmd >> 24) - 0x20);
		tmp += tmp << 1;
		tmp_addr = &cur_seq->t_addr[cur_seq->texture][tmp];
		*tmp_addr = (*tmp_addr & 0x00FFFFFF) | ((cmd & 0xFF) << 24);
		tmp_addr++;
		*tmp_addr = (*tmp_addr & 0x00FFFFFF) | ((cmd & 0xFF00) << 16);
		tmp_addr++;
		*tmp_addr = (*tmp_addr & 0x00FFFFFF) | ((cmd & 0xFF0000) << 8);
		return 0;
	case check_texture_addr2:
		cur_seq->unfinished = tex_address;
		cur_seq->tex_level_lo[tmp = cur_seq->texture] = cmd & 0x3F;
		cur_seq->tex_level_hi[tmp] = (cmd & 0xFC0) >> 6;
		return 0;
	case check_texture_addr3:
		cur_seq->unfinished = tex_address;
		tmp = ((cmd >> 24) - HC_SubA_HTXnL0Pit);
		if (tmp == 0 && 
		    (cmd & HC_HTXnEnPit_MASK)) {
			cur_seq->pitch[cur_seq->texture][tmp] = 
				(cmd & HC_HTXnLnPit_MASK);
			cur_seq->tex_npot[cur_seq->texture] = 1;
		} else {
			cur_seq->pitch[cur_seq->texture][tmp] =
				(cmd & HC_HTXnLnPitE_MASK) >> HC_HTXnLnPitE_SHIFT;
			cur_seq->tex_npot[cur_seq->texture] = 0;			
			if (cmd & 0x000FFFFF) {
				DRM_ERROR
					("Unimplemented texture level 0 pitch mode.\n");
				return 2;
			}
		}
		return 0;
	case check_texture_addr4:
		cur_seq->unfinished = tex_address;
		tmp_addr = &cur_seq->t_addr[cur_seq->texture][9];
		*tmp_addr = (*tmp_addr & 0x00FFFFFF) | ((cmd & 0xFF) << 24);
		return 0;
	case check_texture_addr5:
	case check_texture_addr6:
		cur_seq->unfinished = tex_address;
		/*
		 * Texture width. We don't care since we have the pitch.
		 */
		return 0;
	case check_texture_addr7:
		cur_seq->unfinished = tex_address;
		tmp_addr = &(cur_seq->height[cur_seq->texture][0]);
		tmp_addr[5] = 1 << ((cmd & 0x00F00000) >> 20);
		tmp_addr[4] = 1 << ((cmd & 0x000F0000) >> 16);
		tmp_addr[3] = 1 << ((cmd & 0x0000F000) >> 12);
		tmp_addr[2] = 1 << ((cmd & 0x00000F00) >> 8);
		tmp_addr[1] = 1 << ((cmd & 0x000000F0) >> 4);
		tmp_addr[0] = 1 << (cmd & 0x0000000F);
		return 0;
	case check_texture_addr8:
		cur_seq->unfinished = tex_address;
		tmp_addr = &(cur_seq->height[cur_seq->texture][0]);
		tmp_addr[9] = 1 << ((cmd & 0x0000F000) >> 12);
		tmp_addr[8] = 1 << ((cmd & 0x00000F00) >> 8);
		tmp_addr[7] = 1 << ((cmd & 0x000000F0) >> 4);
		tmp_addr[6] = 1 << (cmd & 0x0000000F);
		return 0;
	case check_texture_addr_mode:
		cur_seq->unfinished = tex_address;
		if (2 == (tmp = cmd & 0x00000003)) {
			DRM_ERROR
			    ("Attempt to fetch texture from system memory.\n");
			return 2;
		}
		cur_seq->agp_texture = (tmp == 3);
		cur_seq->tex_palette_size[cur_seq->texture] =
		    (cmd >> 16) & 0x000000007;
		return 0;
	case check_for_vertex_count:
		cur_seq->vertex_count = cmd & 0x0000FFFF;
		return 0;
	case check_number_texunits:
		cur_seq->multitex = (cmd >> 3) & 1;
		return 0;
	default:
		DRM_ERROR("Illegal DMA data: 0x%x\n", cmd);
		return 2;
	}
	return 2;
}

static __inline__ int
via_check_prim_list(uint32_t const **buffer, const uint32_t * buf_end,
		    drm_via_state_t * cur_seq)
{
	drm_via_private_t *dev_priv =
	    (drm_via_private_t *) cur_seq->dev->dev_private;
	uint32_t a_fire, bcmd, dw_count;
	int ret = 0;
	int have_fire;
	const uint32_t *buf = *buffer;

	while (buf < buf_end) {
		have_fire = 0;
		if ((buf_end - buf) < 2) {
			DRM_ERROR
			    ("Unexpected termination of primitive list.\n");
			ret = 1;
			break;
		}
		if ((*buf & HC_ACMD_MASK) != HC_ACMD_HCmdB)
			break;
		bcmd = *buf++;
		if ((*buf & HC_ACMD_MASK) != HC_ACMD_HCmdA) {
			DRM_ERROR("Expected Vertex List A command, got 0x%x\n",
				  *buf);
			ret = 1;
			break;
		}
		a_fire =
		    *buf++ | HC_HPLEND_MASK | HC_HPMValidN_MASK |
		    HC_HE3Fire_MASK;

		/*
		 * How many dwords per vertex ?
		 */

		if (cur_seq->agp && ((bcmd & (0xF << 11)) == 0)) {
			DRM_ERROR("Illegal B command vertex data for AGP.\n");
			ret = 1;
			break;
		}

		dw_count = 0;
		if (bcmd & (1 << 7))
			dw_count += (cur_seq->multitex) ? 2 : 1;
		if (bcmd & (1 << 8))
			dw_count += (cur_seq->multitex) ? 2 : 1;
		if (bcmd & (1 << 9))
			dw_count++;
		if (bcmd & (1 << 10))
			dw_count++;
		if (bcmd & (1 << 11))
			dw_count++;
		if (bcmd & (1 << 12))
			dw_count++;
		if (bcmd & (1 << 13))
			dw_count++;
		if (bcmd & (1 << 14))
			dw_count++;

		while (buf < buf_end) {
			if (*buf == a_fire) {
				if (dev_priv->num_fire_offsets >=
				    VIA_FIRE_BUF_SIZE) {
					DRM_ERROR("Fire offset buffer full.\n");
					ret = 1;
					break;
				}
				dev_priv->fire_offsets[dev_priv->
						       num_fire_offsets++] =
				    buf;
				have_fire = 1;
				buf++;
				if (buf < buf_end && *buf == a_fire)
					buf++;
				break;
			}
			if ((*buf == HALCYON_HEADER2) ||
			    ((*buf & HALCYON_FIREMASK) == HALCYON_FIRECMD)) {
				DRM_ERROR("Missing Vertex Fire command, "
					  "Stray Vertex Fire command  or verifier "
					  "lost sync.\n");
				ret = 1;
				break;
			}
			if ((ret = eat_words(&buf, buf_end, dw_count)))
				break;
		}
		if (buf >= buf_end && !have_fire) {
			DRM_ERROR("Missing Vertex Fire command or verifier "
				  "lost sync.\n");
			ret = 1;
			break;
		}
		if (cur_seq->agp && ((buf - cur_seq->buf_start) & 0x01)) {
			DRM_ERROR("AGP Primitive list end misaligned.\n");
			ret = 1;
			break;
		}
	}
	*buffer = buf;
	return ret;
}

static __inline__ verifier_state_t
via_check_header2(uint32_t const **buffer, const uint32_t * buf_end,
		  drm_via_state_t * hc_state)
{
	uint32_t cmd;
	int hz_mode;
	hazard_t hz;
	const uint32_t *buf = *buffer;
	const hazard_t *hz_table;

	if ((buf_end - buf) < 2) {
		DRM_ERROR
		    ("Illegal termination of DMA HALCYON_HEADER2 sequence.\n");
		return state_error;
	}
	buf++;
	cmd = (*buf++ & 0xFFFF0000) >> 16;

	switch (cmd) {
	case HC_ParaType_CmdVdata:
		if (via_check_prim_list(&buf, buf_end, hc_state))
			return state_error;
		*buffer = buf;
		return state_command;
	case HC_ParaType_NotTex:
		hz_table = table1;
		break;
	case HC_ParaType_Tex:
		hc_state->texture = 0;
		hz_table = table2;
		break;
	case (HC_ParaType_Tex | (HC_SubType_Tex1 << 8)):
		hc_state->texture = 1;
		hz_table = table2;
		break;
	case (HC_ParaType_Tex | (HC_SubType_TexGeneral << 8)):
		hz_table = table3;
		break;
	case HC_ParaType_Auto:
		if (eat_words(&buf, buf_end, 2))
			return state_error;
		*buffer = buf;
		return state_command;
	case (HC_ParaType_Palette | (HC_SubType_Stipple << 8)):
		if (eat_words(&buf, buf_end, 32))
			return state_error;
		*buffer = buf;
		return state_command;
	case (HC_ParaType_Palette | (HC_SubType_TexPalette0 << 8)):
	case (HC_ParaType_Palette | (HC_SubType_TexPalette1 << 8)):
		DRM_ERROR("Texture palettes are rejected because of "
			  "lack of info how to determine their size.\n");
		return state_error;
	case (HC_ParaType_Palette | (HC_SubType_FogTable << 8)):
		DRM_ERROR("Fog factor palettes are rejected because of "
			  "lack of info how to determine their size.\n");
		return state_error;
	default:

		/*
		 * There are some unimplemented HC_ParaTypes here, that
		 * need to be implemented if the Mesa driver is extended.
		 */

		DRM_ERROR("Invalid or unimplemented HALCYON_HEADER2 "
			  "DMA subcommand: 0x%x. Previous dword: 0x%x\n",
			  cmd, *(buf - 2));
		*buffer = buf;
		return state_error;
	}

	while (buf < buf_end) {
		cmd = *buf++;
		if ((hz = hz_table[cmd >> 24])) {
			if ((hz_mode = investigate_hazard(cmd, hz, hc_state))) {
				if (hz_mode == 1) {
					buf--;
					break;
				}
				return state_error;
			}
		} else if (hc_state->unfinished &&
			   finish_current_sequence(hc_state)) {
			return state_error;
		}
	}
	if (hc_state->unfinished && finish_current_sequence(hc_state)) {
		return state_error;
	}
	*buffer = buf;
	return state_command;
}

static __inline__ verifier_state_t
via_parse_header2(drm_via_private_t * dev_priv, uint32_t const **buffer,
		  const uint32_t * buf_end, int *fire_count)
{
	uint32_t cmd;
	const uint32_t *buf = *buffer;
	const uint32_t *next_fire;
	int burst = 0;

	next_fire = dev_priv->fire_offsets[*fire_count];
	buf++;
	cmd = (*buf & 0xFFFF0000) >> 16;
	VIA_WRITE(HC_REG_TRANS_SET + HC_REG_BASE, *buf++);
	switch (cmd) {
	case HC_ParaType_CmdVdata:
		while ((buf < buf_end) &&
		       (*fire_count < dev_priv->num_fire_offsets) &&
		       (*buf & HC_ACMD_MASK) == HC_ACMD_HCmdB) {
			while (buf <= next_fire) {
				VIA_WRITE(HC_REG_TRANS_SPACE + HC_REG_BASE +
					  (burst & 63), *buf++);
				burst += 4;
			}
			if ((buf < buf_end)
			    && ((*buf & HALCYON_FIREMASK) == HALCYON_FIRECMD))
				buf++;

			if (++(*fire_count) < dev_priv->num_fire_offsets)
				next_fire = dev_priv->fire_offsets[*fire_count];
		}
		break;
	default:
		while (buf < buf_end) {

			if (*buf == HC_HEADER2 ||
			    (*buf & HALCYON_HEADER1MASK) == HALCYON_HEADER1 ||
			    (*buf & VIA_VIDEOMASK) == VIA_VIDEO_HEADER5 ||
			    (*buf & VIA_VIDEOMASK) == VIA_VIDEO_HEADER6)
				break;

			VIA_WRITE(HC_REG_TRANS_SPACE + HC_REG_BASE +
				  (burst & 63), *buf++);
			burst += 4;
		}
	}
	*buffer = buf;
	return state_command;
}

static __inline__ int verify_mmio_address(uint32_t address)
{
	if ((address > 0x3FF) && (address < 0xC00)) {
		DRM_ERROR("Invalid VIDEO DMA command. "
			  "Attempt to access 3D- or command burst area.\n");
		return 1;
	} else if ((address > 0xCFF) && (address < 0x1300)) {
		DRM_ERROR("Invalid VIDEO DMA command. "
			  "Attempt to access PCI DMA area.\n");
		return 1;
	} else if (address > 0x13FF) {
		DRM_ERROR("Invalid VIDEO DMA command. "
			  "Attempt to access VGA registers.\n");
		return 1;
	}
	return 0;
}

static __inline__ int
verify_video_tail(uint32_t const **buffer, const uint32_t * buf_end,
		  uint32_t dwords)
{
	const uint32_t *buf = *buffer;

	if (buf_end - buf < dwords) {
		DRM_ERROR("Illegal termination of video command.\n");
		return 1;
	}
	while (dwords--) {
		if (*buf++) {
			DRM_ERROR("Illegal video command tail.\n");
			return 1;
		}
	}
	*buffer = buf;
	return 0;
}

static __inline__ verifier_state_t
via_check_header1(uint32_t const **buffer, const uint32_t * buf_end)
{
	uint32_t cmd;
	const uint32_t *buf = *buffer;
	verifier_state_t ret = state_command;

	while (buf < buf_end) {
		cmd = *buf;
		if ((cmd > ((0x3FF >> 2) | HALCYON_HEADER1)) &&
		    (cmd < ((0xC00 >> 2) | HALCYON_HEADER1))) {
			if ((cmd & HALCYON_HEADER1MASK) != HALCYON_HEADER1)
				break;
			DRM_ERROR("Invalid HALCYON_HEADER1 command. "
				  "Attempt to access 3D- or command burst area.\n");
			ret = state_error;
			break;
		} else if (cmd > ((0xCFF >> 2) | HALCYON_HEADER1)) {
			if ((cmd & HALCYON_HEADER1MASK) != HALCYON_HEADER1)
				break;
			DRM_ERROR("Invalid HALCYON_HEADER1 command. "
				  "Attempt to access VGA registers.\n");
			ret = state_error;
			break;
		} else {
			buf += 2;
		}
	}
	*buffer = buf;
	return ret;
}

static __inline__ verifier_state_t
via_parse_header1(drm_via_private_t * dev_priv, uint32_t const **buffer,
		  const uint32_t * buf_end)
{
	register uint32_t cmd;
	const uint32_t *buf = *buffer;

	while (buf < buf_end) {
		cmd = *buf;
		if ((cmd & HALCYON_HEADER1MASK) != HALCYON_HEADER1)
			break;
		VIA_WRITE((cmd & ~HALCYON_HEADER1MASK) << 2, *++buf);
		buf++;
	}
	*buffer = buf;
	return state_command;
}

static __inline__ verifier_state_t
via_check_vheader5(uint32_t const **buffer, const uint32_t * buf_end)
{
	uint32_t data;
	const uint32_t *buf = *buffer;

	if (buf_end - buf < 4) {
		DRM_ERROR("Illegal termination of video header5 command\n");
		return state_error;
	}

	data = *buf++ & ~VIA_VIDEOMASK;
	if (verify_mmio_address(data))
		return state_error;

	data = *buf++;
	if (*buf++ != 0x00F50000) {
		DRM_ERROR("Illegal header5 header data\n");
		return state_error;
	}
	if (*buf++ != 0x00000000) {
		DRM_ERROR("Illegal header5 header data\n");
		return state_error;
	}
	if (eat_words(&buf, buf_end, data))
		return state_error;
	if ((data & 3) && verify_video_tail(&buf, buf_end, 4 - (data & 3)))
		return state_error;
	*buffer = buf;
	return state_command;

}

static __inline__ verifier_state_t
via_parse_vheader5(drm_via_private_t * dev_priv, uint32_t const **buffer,
		   const uint32_t * buf_end)
{
	uint32_t addr, count, i;
	const uint32_t *buf = *buffer;

	addr = *buf++ & ~VIA_VIDEOMASK;
	i = count = *buf;
	buf += 3;
	while (i--) {
		VIA_WRITE(addr, *buf++);
	}
	if (count & 3)
		buf += 4 - (count & 3);
	*buffer = buf;
	return state_command;
}

static __inline__ verifier_state_t
via_check_vheader6(uint32_t const **buffer, const uint32_t * buf_end)
{
	uint32_t data;
	const uint32_t *buf = *buffer;
	uint32_t i;

	if (buf_end - buf < 4) {
		DRM_ERROR("Illegal termination of video header6 command\n");
		return state_error;
	}
	buf++;
	data = *buf++;
	if (*buf++ != 0x00F60000) {
		DRM_ERROR("Illegal header6 header data\n");
		return state_error;
	}
	if (*buf++ != 0x00000000) {
		DRM_ERROR("Illegal header6 header data\n");
		return state_error;
	}
	if ((buf_end - buf) < (data << 1)) {
		DRM_ERROR("Illegal termination of video header6 command\n");
		return state_error;
	}
	for (i = 0; i < data; ++i) {
		if (verify_mmio_address(*buf++))
			return state_error;
		buf++;
	}
	data <<= 1;
	if ((data & 3) && verify_video_tail(&buf, buf_end, 4 - (data & 3)))
		return state_error;
	*buffer = buf;
	return state_command;
}

static __inline__ verifier_state_t
via_parse_vheader6(drm_via_private_t * dev_priv, uint32_t const **buffer,
		   const uint32_t * buf_end)
{

	uint32_t addr, count, i;
	const uint32_t *buf = *buffer;

	i = count = *++buf;
	buf += 3;
	while (i--) {
		addr = *buf++;
		VIA_WRITE(addr, *buf++);
	}
	count <<= 1;
	if (count & 3)
		buf += 4 - (count & 3);
	*buffer = buf;
	return state_command;
}

int
via_verify_command_stream(const uint32_t * buf, unsigned int size,
			  drm_device_t * dev, int agp)
{

	drm_via_private_t *dev_priv = (drm_via_private_t *) dev->dev_private;
	drm_via_state_t *hc_state = &dev_priv->hc_state;
	drm_via_state_t saved_state = *hc_state;
	uint32_t cmd;
	const uint32_t *buf_end = buf + (size >> 2);
	verifier_state_t state = state_command;
	int cme_video;
	int supported_3d;

	cme_video = (dev_priv->chipset == VIA_PRO_GROUP_A ||
		     dev_priv->chipset == VIA_DX9_0);

	supported_3d = dev_priv->chipset != VIA_DX9_0;

	hc_state->dev = dev;
	hc_state->unfinished = no_sequence;
	hc_state->map_cache = NULL;
	hc_state->agp = agp;
	hc_state->buf_start = buf;
	dev_priv->num_fire_offsets = 0;

	while (buf < buf_end) {

		switch (state) {
		case state_header2:
			state = via_check_header2(&buf, buf_end, hc_state);
			break;
		case state_header1:
			state = via_check_header1(&buf, buf_end);
			break;
		case state_vheader5:
			state = via_check_vheader5(&buf, buf_end);
			break;
		case state_vheader6:
			state = via_check_vheader6(&buf, buf_end);
			break;
		case state_command:
			if ((HALCYON_HEADER2 == (cmd = *buf)) && 
			    supported_3d)
				state = state_header2;
			else if ((cmd & HALCYON_HEADER1MASK) == HALCYON_HEADER1)
				state = state_header1;
			else if (cme_video
				 && (cmd & VIA_VIDEOMASK) == VIA_VIDEO_HEADER5)
				state = state_vheader5;
			else if (cme_video
				 && (cmd & VIA_VIDEOMASK) == VIA_VIDEO_HEADER6)
				state = state_vheader6;
			else if ((cmd == HALCYON_HEADER2) && !supported_3d) {
				DRM_ERROR("Accelerated 3D is not supported on this chipset yet.\n");
				state = state_error;
			} else {
				DRM_ERROR
				    ("Invalid / Unimplemented DMA HEADER command. 0x%x\n",
				     cmd);
				state = state_error;
			}
			break;
		case state_error:
		default:
			*hc_state = saved_state;
			return DRM_ERR(EINVAL);
		}
	}
	if (state == state_error) {
		*hc_state = saved_state;
		return DRM_ERR(EINVAL);
	}
	return 0;