2 * Copyright 1998-2008 VIA Technologies, Inc. All Rights Reserved.
3 * Copyright 2001-2008 S3 Graphics, Inc. All Rights Reserved.
4 * Copyright 2011 Florian Tobias Schandinat <FlorianSchandinat@gmx.de>
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public
8 * License as published by the Free Software Foundation;
9 * either version 2, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTIES OR REPRESENTATIONS; without even
13 * the implied warranty of MERCHANTABILITY or FITNESS FOR
14 * A PARTICULAR PURPOSE.See the GNU General Public License
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
20 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23 * clock and PLL management functions
26 #include <linux/kernel.h>
27 #include <linux/via-core.h>
28 #include "via_clock.h"
32 const char *via_slap
= "Please slap VIA Technologies to motivate them "
33 "releasing full documentation for your platform!\n";
35 static inline u32
cle266_encode_pll(struct via_pll_config pll
)
37 return (pll
.multiplier
<< 8)
42 static inline u32
k800_encode_pll(struct via_pll_config pll
)
44 return ((pll
.divisor
- 2) << 16)
46 | (pll
.multiplier
- 2);
49 static inline u32
vx855_encode_pll(struct via_pll_config pll
)
51 return (pll
.divisor
<< 16)
56 static inline void cle266_set_primary_pll_encoded(u32 data
)
58 via_write_reg_mask(VIASR
, 0x40, 0x02, 0x02); /* enable reset */
59 via_write_reg(VIASR
, 0x46, data
& 0xFF);
60 via_write_reg(VIASR
, 0x47, (data
>> 8) & 0xFF);
61 via_write_reg_mask(VIASR
, 0x40, 0x00, 0x02); /* disable reset */
64 static inline void k800_set_primary_pll_encoded(u32 data
)
66 via_write_reg_mask(VIASR
, 0x40, 0x02, 0x02); /* enable reset */
67 via_write_reg(VIASR
, 0x44, data
& 0xFF);
68 via_write_reg(VIASR
, 0x45, (data
>> 8) & 0xFF);
69 via_write_reg(VIASR
, 0x46, (data
>> 16) & 0xFF);
70 via_write_reg_mask(VIASR
, 0x40, 0x00, 0x02); /* disable reset */
73 static inline void cle266_set_secondary_pll_encoded(u32 data
)
75 via_write_reg_mask(VIASR
, 0x40, 0x04, 0x04); /* enable reset */
76 via_write_reg(VIASR
, 0x44, data
& 0xFF);
77 via_write_reg(VIASR
, 0x45, (data
>> 8) & 0xFF);
78 via_write_reg_mask(VIASR
, 0x40, 0x00, 0x04); /* disable reset */
81 static inline void k800_set_secondary_pll_encoded(u32 data
)
83 via_write_reg_mask(VIASR
, 0x40, 0x04, 0x04); /* enable reset */
84 via_write_reg(VIASR
, 0x4A, data
& 0xFF);
85 via_write_reg(VIASR
, 0x4B, (data
>> 8) & 0xFF);
86 via_write_reg(VIASR
, 0x4C, (data
>> 16) & 0xFF);
87 via_write_reg_mask(VIASR
, 0x40, 0x00, 0x04); /* disable reset */
90 static inline void set_engine_pll_encoded(u32 data
)
92 via_write_reg_mask(VIASR
, 0x40, 0x01, 0x01); /* enable reset */
93 via_write_reg(VIASR
, 0x47, data
& 0xFF);
94 via_write_reg(VIASR
, 0x48, (data
>> 8) & 0xFF);
95 via_write_reg(VIASR
, 0x49, (data
>> 16) & 0xFF);
96 via_write_reg_mask(VIASR
, 0x40, 0x00, 0x01); /* disable reset */
99 static void cle266_set_primary_pll(struct via_pll_config config
)
101 cle266_set_primary_pll_encoded(cle266_encode_pll(config
));
104 static void k800_set_primary_pll(struct via_pll_config config
)
106 k800_set_primary_pll_encoded(k800_encode_pll(config
));
109 static void vx855_set_primary_pll(struct via_pll_config config
)
111 k800_set_primary_pll_encoded(vx855_encode_pll(config
));
114 static void cle266_set_secondary_pll(struct via_pll_config config
)
116 cle266_set_secondary_pll_encoded(cle266_encode_pll(config
));
119 static void k800_set_secondary_pll(struct via_pll_config config
)
121 k800_set_secondary_pll_encoded(k800_encode_pll(config
));
124 static void vx855_set_secondary_pll(struct via_pll_config config
)
126 k800_set_secondary_pll_encoded(vx855_encode_pll(config
));
129 static void k800_set_engine_pll(struct via_pll_config config
)
131 set_engine_pll_encoded(k800_encode_pll(config
));
134 static void vx855_set_engine_pll(struct via_pll_config config
)
136 set_engine_pll_encoded(vx855_encode_pll(config
));
139 static void set_primary_pll_state(u8 state
)
154 via_write_reg_mask(VIASR
, 0x2D, value
, 0x30);
157 static void set_secondary_pll_state(u8 state
)
172 via_write_reg_mask(VIASR
, 0x2D, value
, 0x0C);
175 static void set_engine_pll_state(u8 state
)
190 via_write_reg_mask(VIASR
, 0x2D, value
, 0x03);
193 static void set_primary_clock_state(u8 state
)
208 via_write_reg_mask(VIASR
, 0x1B, value
, 0x30);
211 static void set_secondary_clock_state(u8 state
)
226 via_write_reg_mask(VIASR
, 0x1B, value
, 0xC0);
229 static inline u8
set_clock_source_common(enum via_clksrc source
, bool use_pll
)
237 case VIA_CLKSRC_TVX1
:
240 case VIA_CLKSRC_TVPLL
:
241 data
= 0x04; /* 0x06 should be the same */
243 case VIA_CLKSRC_DVP1TVCLKR
:
246 case VIA_CLKSRC_CAP0
:
249 case VIA_CLKSRC_CAP1
:
260 static void set_primary_clock_source(enum via_clksrc source
, bool use_pll
)
262 u8 data
= set_clock_source_common(source
, use_pll
) << 4;
263 via_write_reg_mask(VIACR
, 0x6C, data
, 0xF0);
266 static void set_secondary_clock_source(enum via_clksrc source
, bool use_pll
)
268 u8 data
= set_clock_source_common(source
, use_pll
);
269 via_write_reg_mask(VIACR
, 0x6C, data
, 0x0F);
272 static void dummy_set_clock_state(u8 state
)
274 printk(KERN_INFO
"Using undocumented set clock state.\n%s", via_slap
);
277 static void dummy_set_clock_source(enum via_clksrc source
, bool use_pll
)
279 printk(KERN_INFO
"Using undocumented set clock source.\n%s", via_slap
);
282 static void dummy_set_pll_state(u8 state
)
284 printk(KERN_INFO
"Using undocumented set PLL state.\n%s", via_slap
);
287 static void dummy_set_pll(struct via_pll_config config
)
289 printk(KERN_INFO
"Using undocumented set PLL.\n%s", via_slap
);
292 void via_clock_init(struct via_clock
*clock
, int gfx_chip
)
295 case UNICHROME_CLE266
:
297 clock
->set_primary_clock_state
= dummy_set_clock_state
;
298 clock
->set_primary_clock_source
= dummy_set_clock_source
;
299 clock
->set_primary_pll_state
= dummy_set_pll_state
;
300 clock
->set_primary_pll
= cle266_set_primary_pll
;
302 clock
->set_secondary_clock_state
= dummy_set_clock_state
;
303 clock
->set_secondary_clock_source
= dummy_set_clock_source
;
304 clock
->set_secondary_pll_state
= dummy_set_pll_state
;
305 clock
->set_secondary_pll
= cle266_set_secondary_pll
;
307 clock
->set_engine_pll_state
= dummy_set_pll_state
;
308 clock
->set_engine_pll
= dummy_set_pll
;
311 case UNICHROME_PM800
:
312 case UNICHROME_CN700
:
313 case UNICHROME_CX700
:
314 case UNICHROME_CN750
:
315 case UNICHROME_K8M890
:
316 case UNICHROME_P4M890
:
317 case UNICHROME_P4M900
:
318 case UNICHROME_VX800
:
319 clock
->set_primary_clock_state
= set_primary_clock_state
;
320 clock
->set_primary_clock_source
= set_primary_clock_source
;
321 clock
->set_primary_pll_state
= set_primary_pll_state
;
322 clock
->set_primary_pll
= k800_set_primary_pll
;
324 clock
->set_secondary_clock_state
= set_secondary_clock_state
;
325 clock
->set_secondary_clock_source
= set_secondary_clock_source
;
326 clock
->set_secondary_pll_state
= set_secondary_pll_state
;
327 clock
->set_secondary_pll
= k800_set_secondary_pll
;
329 clock
->set_engine_pll_state
= set_engine_pll_state
;
330 clock
->set_engine_pll
= k800_set_engine_pll
;
332 case UNICHROME_VX855
:
333 case UNICHROME_VX900
:
334 clock
->set_primary_clock_state
= set_primary_clock_state
;
335 clock
->set_primary_clock_source
= set_primary_clock_source
;
336 clock
->set_primary_pll_state
= set_primary_pll_state
;
337 clock
->set_primary_pll
= vx855_set_primary_pll
;
339 clock
->set_secondary_clock_state
= set_secondary_clock_state
;
340 clock
->set_secondary_clock_source
= set_secondary_clock_source
;
341 clock
->set_secondary_pll_state
= set_secondary_pll_state
;
342 clock
->set_secondary_pll
= vx855_set_secondary_pll
;
344 clock
->set_engine_pll_state
= set_engine_pll_state
;
345 clock
->set_engine_pll
= vx855_set_engine_pll
;