This source file includes following definitions.
- __do_cyrix_devid
- do_cyrix_devid
- check_cx686_slop
- set_cx86_reorder
- set_cx86_memwb
- geode_configure
- early_init_cyrix
- init_cyrix
- init_nsc
- test_cyrix_52div
- cyrix_identify
1
2 #include <linux/bitops.h>
3 #include <linux/delay.h>
4 #include <linux/pci.h>
5 #include <asm/dma.h>
6 #include <linux/io.h>
7 #include <asm/processor-cyrix.h>
8 #include <asm/processor-flags.h>
9 #include <linux/timer.h>
10 #include <asm/pci-direct.h>
11 #include <asm/tsc.h>
12 #include <asm/cpufeature.h>
13 #include <linux/sched.h>
14 #include <linux/sched/clock.h>
15
16 #include "cpu.h"
17
18
19
20
21 static void __do_cyrix_devid(unsigned char *dir0, unsigned char *dir1)
22 {
23 unsigned char ccr2, ccr3;
24
25
26 ccr3 = getCx86(CX86_CCR3);
27 setCx86(CX86_CCR3, ccr3 ^ 0x80);
28 getCx86(0xc0);
29
30 if (getCx86(CX86_CCR3) == ccr3) {
31 ccr2 = getCx86(CX86_CCR2);
32 setCx86(CX86_CCR2, ccr2 ^ 0x04);
33 getCx86(0xc0);
34
35 if (getCx86(CX86_CCR2) == ccr2)
36 *dir0 = 0xfd;
37 else {
38 setCx86(CX86_CCR2, ccr2);
39 *dir0 = 0xfe;
40 }
41 } else {
42 setCx86(CX86_CCR3, ccr3);
43
44
45 *dir0 = getCx86(CX86_DIR0);
46 *dir1 = getCx86(CX86_DIR1);
47 }
48 }
49
50 static void do_cyrix_devid(unsigned char *dir0, unsigned char *dir1)
51 {
52 unsigned long flags;
53
54 local_irq_save(flags);
55 __do_cyrix_devid(dir0, dir1);
56 local_irq_restore(flags);
57 }
58
59
60
61
62
63
64
65 static unsigned char Cx86_dir0_msb = 0;
66
67 static const char Cx86_model[][9] = {
68 "Cx486", "Cx486", "5x86 ", "6x86", "MediaGX ", "6x86MX ",
69 "M II ", "Unknown"
70 };
71 static const char Cx486_name[][5] = {
72 "SLC", "DLC", "SLC2", "DLC2", "SRx", "DRx",
73 "SRx2", "DRx2"
74 };
75 static const char Cx486S_name[][4] = {
76 "S", "S2", "Se", "S2e"
77 };
78 static const char Cx486D_name[][4] = {
79 "DX", "DX2", "?", "?", "?", "DX4"
80 };
81 static char Cx86_cb[] = "?.5x Core/Bus Clock";
82 static const char cyrix_model_mult1[] = "12??43";
83 static const char cyrix_model_mult2[] = "12233445";
84
85
86
87
88
89
90
91
92
93 static void check_cx686_slop(struct cpuinfo_x86 *c)
94 {
95 unsigned long flags;
96
97 if (Cx86_dir0_msb == 3) {
98 unsigned char ccr3, ccr5;
99
100 local_irq_save(flags);
101 ccr3 = getCx86(CX86_CCR3);
102 setCx86(CX86_CCR3, (ccr3 & 0x0f) | 0x10);
103 ccr5 = getCx86(CX86_CCR5);
104 if (ccr5 & 2)
105 setCx86(CX86_CCR5, ccr5 & 0xfd);
106 setCx86(CX86_CCR3, ccr3);
107 local_irq_restore(flags);
108
109 if (ccr5 & 2) {
110 pr_info("Recalibrating delay loop with SLOP bit reset\n");
111 calibrate_delay();
112 c->loops_per_jiffy = loops_per_jiffy;
113 }
114 }
115 }
116
117
118 static void set_cx86_reorder(void)
119 {
120 u8 ccr3;
121
122 pr_info("Enable Memory access reorder on Cyrix/NSC processor.\n");
123 ccr3 = getCx86(CX86_CCR3);
124 setCx86(CX86_CCR3, (ccr3 & 0x0f) | 0x10);
125
126
127 setCx86(CX86_PCR0, getCx86(CX86_PCR0) & ~0x80);
128
129 ccr3 |= 0xe0;
130 setCx86(CX86_CCR3, ccr3);
131 }
132
133 static void set_cx86_memwb(void)
134 {
135 pr_info("Enable Memory-Write-back mode on Cyrix/NSC processor.\n");
136
137
138 setCx86(CX86_CCR2, getCx86(CX86_CCR2) & ~0x04);
139
140 write_cr0(read_cr0() | X86_CR0_NW);
141
142 setCx86(CX86_CCR2, getCx86(CX86_CCR2) | 0x14);
143 }
144
145
146
147
148
149 static void geode_configure(void)
150 {
151 unsigned long flags;
152 u8 ccr3;
153 local_irq_save(flags);
154
155
156 setCx86(CX86_CCR2, getCx86(CX86_CCR2) | 0x88);
157
158 ccr3 = getCx86(CX86_CCR3);
159 setCx86(CX86_CCR3, (ccr3 & 0x0f) | 0x10);
160
161
162
163 setCx86(CX86_CCR4, getCx86(CX86_CCR4) | 0x38);
164 setCx86(CX86_CCR3, ccr3);
165
166 set_cx86_memwb();
167 set_cx86_reorder();
168
169 local_irq_restore(flags);
170 }
171
172 static void early_init_cyrix(struct cpuinfo_x86 *c)
173 {
174 unsigned char dir0, dir0_msn, dir1 = 0;
175
176 __do_cyrix_devid(&dir0, &dir1);
177 dir0_msn = dir0 >> 4;
178
179 switch (dir0_msn) {
180 case 3:
181
182 set_cpu_cap(c, X86_FEATURE_CYRIX_ARR);
183 break;
184 case 5:
185
186 set_cpu_cap(c, X86_FEATURE_CYRIX_ARR);
187 break;
188 }
189 }
190
191 static void init_cyrix(struct cpuinfo_x86 *c)
192 {
193 unsigned char dir0, dir0_msn, dir0_lsn, dir1 = 0;
194 char *buf = c->x86_model_id;
195 const char *p = NULL;
196
197
198
199
200
201 clear_cpu_cap(c, 0*32+31);
202
203
204 if (test_cpu_cap(c, 1*32+24)) {
205 clear_cpu_cap(c, 1*32+24);
206 set_cpu_cap(c, X86_FEATURE_CXMMX);
207 }
208
209 do_cyrix_devid(&dir0, &dir1);
210
211 check_cx686_slop(c);
212
213 Cx86_dir0_msb = dir0_msn = dir0 >> 4;
214 dir0_lsn = dir0 & 0xf;
215
216
217 c->x86_model = (dir1 >> 4) + 1;
218 c->x86_stepping = dir1 & 0xf;
219
220
221
222
223
224
225
226 switch (dir0_msn) {
227 unsigned char tmp;
228
229 case 0:
230 p = Cx486_name[dir0_lsn & 7];
231 break;
232
233 case 1:
234 p = (dir0_lsn & 8) ? Cx486D_name[dir0_lsn & 5]
235 : Cx486S_name[dir0_lsn & 3];
236 break;
237
238 case 2:
239 Cx86_cb[2] = cyrix_model_mult1[dir0_lsn & 5];
240 p = Cx86_cb+2;
241 break;
242
243 case 3:
244 Cx86_cb[1] = ' ';
245 Cx86_cb[2] = cyrix_model_mult1[dir0_lsn & 5];
246 if (dir1 > 0x21) {
247 Cx86_cb[0] = 'L';
248 p = Cx86_cb;
249 (c->x86_model)++;
250 } else
251 p = Cx86_cb+1;
252
253 set_cpu_cap(c, X86_FEATURE_CYRIX_ARR);
254
255 set_cpu_bug(c, X86_BUG_COMA);
256 break;
257
258 case 4:
259 case 11:
260 #ifdef CONFIG_PCI
261 {
262 u32 vendor, device;
263
264
265
266
267
268
269
270
271
272
273
274
275
276 pr_info("Working around Cyrix MediaGX virtual DMA bugs.\n");
277 isa_dma_bridge_buggy = 2;
278
279
280
281
282 vendor = read_pci_config_16(0, 0, 0x12, PCI_VENDOR_ID);
283 device = read_pci_config_16(0, 0, 0x12, PCI_DEVICE_ID);
284
285
286
287
288 if (vendor == PCI_VENDOR_ID_CYRIX &&
289 (device == PCI_DEVICE_ID_CYRIX_5510 ||
290 device == PCI_DEVICE_ID_CYRIX_5520))
291 mark_tsc_unstable("cyrix 5510/5520 detected");
292 }
293 #endif
294 c->x86_cache_size = 16;
295
296
297 if (c->cpuid_level == 2) {
298
299 setCx86(CX86_CCR7, getCx86(CX86_CCR7) | 1);
300
301
302
303
304
305
306
307 if ((0x30 <= dir1 && dir1 <= 0x6f) ||
308 (0x80 <= dir1 && dir1 <= 0x8f))
309 geode_configure();
310 return;
311 } else {
312 Cx86_cb[2] = (dir0_lsn & 1) ? '3' : '4';
313 p = Cx86_cb+2;
314 c->x86_model = (dir1 & 0x20) ? 1 : 2;
315 }
316 break;
317
318 case 5:
319 if (dir1 > 7) {
320 dir0_msn++;
321
322 setCx86(CX86_CCR7, getCx86(CX86_CCR7)|1);
323 } else {
324
325 set_cpu_bug(c, X86_BUG_COMA);
326 }
327 tmp = (!(dir0_lsn & 7) || dir0_lsn & 1) ? 2 : 0;
328 Cx86_cb[tmp] = cyrix_model_mult2[dir0_lsn & 7];
329 p = Cx86_cb+tmp;
330 if (((dir1 & 0x0f) > 4) || ((dir1 & 0xf0) == 0x20))
331 (c->x86_model)++;
332
333 set_cpu_cap(c, X86_FEATURE_CYRIX_ARR);
334 break;
335
336 case 0xf:
337 switch (dir0_lsn) {
338 case 0xd:
339 dir0_msn = 0;
340 p = Cx486_name[!!boot_cpu_has(X86_FEATURE_FPU)];
341 break;
342
343 case 0xe:
344 dir0_msn = 0;
345 p = Cx486S_name[0];
346 break;
347 }
348 break;
349
350 default:
351 dir0_msn = 7;
352 break;
353 }
354 strcpy(buf, Cx86_model[dir0_msn & 7]);
355 if (p)
356 strcat(buf, p);
357 return;
358 }
359
360
361
362
363 static void init_nsc(struct cpuinfo_x86 *c)
364 {
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380 if (c->x86 == 5 && c->x86_model == 5)
381 cpu_detect_cache_sizes(c);
382 else
383 init_cyrix(c);
384 }
385
386
387
388
389
390
391
392
393
394
395
396 static inline int test_cyrix_52div(void)
397 {
398 unsigned int test;
399
400 __asm__ __volatile__(
401 "sahf\n\t"
402 "div %b2\n\t"
403 "lahf"
404 : "=a" (test)
405 : "0" (5), "q" (2)
406 : "cc");
407
408
409 return (unsigned char) (test >> 8) == 0x02;
410 }
411
412 static void cyrix_identify(struct cpuinfo_x86 *c)
413 {
414
415 if (c->x86 == 4 && test_cyrix_52div()) {
416 unsigned char dir0, dir1;
417
418 strcpy(c->x86_vendor_id, "CyrixInstead");
419 c->x86_vendor = X86_VENDOR_CYRIX;
420
421
422
423
424
425 do_cyrix_devid(&dir0, &dir1);
426
427 dir0 >>= 4;
428
429
430
431 if (dir0 == 5 || dir0 == 3) {
432 unsigned char ccr3;
433 unsigned long flags;
434 pr_info("Enabling CPUID on Cyrix processor.\n");
435 local_irq_save(flags);
436 ccr3 = getCx86(CX86_CCR3);
437
438 setCx86(CX86_CCR3, (ccr3 & 0x0f) | 0x10);
439
440 setCx86(CX86_CCR4, getCx86(CX86_CCR4) | 0x80);
441
442 setCx86(CX86_CCR3, ccr3);
443 local_irq_restore(flags);
444 }
445 }
446 }
447
448 static const struct cpu_dev cyrix_cpu_dev = {
449 .c_vendor = "Cyrix",
450 .c_ident = { "CyrixInstead" },
451 .c_early_init = early_init_cyrix,
452 .c_init = init_cyrix,
453 .c_identify = cyrix_identify,
454 .c_x86_vendor = X86_VENDOR_CYRIX,
455 };
456
457 cpu_dev_register(cyrix_cpu_dev);
458
459 static const struct cpu_dev nsc_cpu_dev = {
460 .c_vendor = "NSC",
461 .c_ident = { "Geode by NSC" },
462 .c_init = init_nsc,
463 .c_x86_vendor = X86_VENDOR_NSC,
464 };
465
466 cpu_dev_register(nsc_cpu_dev);