2 * Copyright 2013 Red Hat Inc.
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
22 * Authors: Ben Skeggs <bskeggs@redhat.com>
26 /* binary driver only executes this path if the condition (a) is true
27 * for any configuration (combination of rammap+ramcfg+timing) that
28 * can be reached on a given card. for now, we will execute the branch
29 * unconditionally in the hope that a "false everywhere" in the bios
30 * tables doesn't actually mean "don't touch this".
35 nvkm_gddr5_calc(struct nvkm_ram
*ram
, bool nuts
)
37 int pd
, lf
, xd
, vh
, vr
, vo
, l3
;
38 int WL
, CL
, WR
, at
[2], dt
, ds
;
39 int rq
= ram
->freq
< 1000000; /* XXX */
41 switch (ram
->next
->bios
.ramcfg_ver
) {
43 pd
= ram
->next
->bios
.ramcfg_11_01_80
;
44 lf
= ram
->next
->bios
.ramcfg_11_01_40
;
45 xd
= !ram
->next
->bios
.ramcfg_11_01_20
;
46 vh
= ram
->next
->bios
.ramcfg_11_02_10
;
47 vr
= ram
->next
->bios
.ramcfg_11_02_04
;
48 vo
= ram
->next
->bios
.ramcfg_11_06
;
49 l3
= !ram
->next
->bios
.ramcfg_11_07_02
;
55 switch (ram
->next
->bios
.timing_ver
) {
57 WL
= (ram
->next
->bios
.timing
[1] & 0x00000f80) >> 7;
58 CL
= (ram
->next
->bios
.timing
[1] & 0x0000001f);
59 WR
= (ram
->next
->bios
.timing
[2] & 0x007f0000) >> 16;
60 at
[0] = ram
->next
->bios
.timing_20_2e_c0
;
61 at
[1] = ram
->next
->bios
.timing_20_2e_30
;
62 dt
= ram
->next
->bios
.timing_20_2e_03
;
63 ds
= ram
->next
->bios
.timing_20_2f_03
;
69 if (WL
< 1 || WL
> 7 || CL
< 5 || CL
> 36 || WR
< 4 || WR
> 35)
75 ram
->mr
[0] |= (WR
& 0x0f) << 8;
76 ram
->mr
[0] |= (CL
& 0x0f) << 3;
77 ram
->mr
[0] |= (WL
& 0x07) << 0;
80 ram
->mr
[1] |= (xd
& 0x01) << 7;
81 ram
->mr
[1] |= (at
[0] & 0x03) << 4;
82 ram
->mr
[1] |= (dt
& 0x03) << 2;
83 ram
->mr
[1] |= (ds
& 0x03) << 0;
85 /* this seems wrong, alternate field used for the broadcast
86 * on nuts vs non-nuts configs.. meh, it matches for now.
88 ram
->mr1_nuts
= ram
->mr
[1];
91 ram
->mr
[1] |= (at
[1] & 0x03) << 4;
95 ram
->mr
[3] |= (rq
& 0x01) << 5;
98 ram
->mr
[5] |= (l3
<< 2);
101 vo
= (ram
->mr
[6] & 0xff0) >> 4;
102 if (ram
->mr
[6] & 0x001)
103 pd
= 1; /* binary driver does this.. bug? */
104 ram
->mr
[6] &= ~0xff1;
105 ram
->mr
[6] |= (vo
& 0xff) << 4;
106 ram
->mr
[6] |= (pd
& 0x01) << 0;
109 ram
->mr
[7] &= ~0x300;
110 ram
->mr
[7] |= (vr
& 0x03) << 8;
112 ram
->mr
[7] &= ~0x088;
113 ram
->mr
[7] |= (vh
& 0x01) << 7;
114 ram
->mr
[7] |= (lf
& 0x01) << 3;
116 ram
->mr
[8] &= ~0x003;
117 ram
->mr
[8] |= (WR
& 0x10) >> 3;
118 ram
->mr
[8] |= (CL
& 0x10) >> 4;