xref: /linux/drivers/scsi/sym53c8xx_2/sym_fw.c (revision f3d9478b2ce468c3115b02ecae7e975990697f15)
1 /*
2  * Device driver for the SYMBIOS/LSILOGIC 53C8XX and 53C1010 family
3  * of PCI-SCSI IO processors.
4  *
5  * Copyright (C) 1999-2001  Gerard Roudier <groudier@free.fr>
6  *
7  * This driver is derived from the Linux sym53c8xx driver.
8  * Copyright (C) 1998-2000  Gerard Roudier
9  *
10  * The sym53c8xx driver is derived from the ncr53c8xx driver that had been
11  * a port of the FreeBSD ncr driver to Linux-1.2.13.
12  *
13  * The original ncr driver has been written for 386bsd and FreeBSD by
14  *         Wolfgang Stanglmeier        <wolf@cologne.de>
15  *         Stefan Esser                <se@mi.Uni-Koeln.de>
16  * Copyright (C) 1994  Wolfgang Stanglmeier
17  *
18  * Other major contributions:
19  *
20  * NVRAM detection and reading.
21  * Copyright (C) 1997 Richard Waltham <dormouse@farsrobt.demon.co.uk>
22  *
23  *-----------------------------------------------------------------------------
24  *
25  * This program is free software; you can redistribute it and/or modify
26  * it under the terms of the GNU General Public License as published by
27  * the Free Software Foundation; either version 2 of the License, or
28  * (at your option) any later version.
29  *
30  * This program is distributed in the hope that it will be useful,
31  * but WITHOUT ANY WARRANTY; without even the implied warranty of
32  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
33  * GNU General Public License for more details.
34  *
35  * You should have received a copy of the GNU General Public License
36  * along with this program; if not, write to the Free Software
37  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
38  */
39 
40 #include "sym_glue.h"
41 
42 /*
43  *  Macros used for all firmwares.
44  */
45 #define	SYM_GEN_A(s, label)	((short) offsetof(s, label)),
46 #define	SYM_GEN_B(s, label)	((short) offsetof(s, label)),
47 #define	SYM_GEN_Z(s, label)	((short) offsetof(s, label)),
48 #define	PADDR_A(label)		SYM_GEN_PADDR_A(struct SYM_FWA_SCR, label)
49 #define	PADDR_B(label)		SYM_GEN_PADDR_B(struct SYM_FWB_SCR, label)
50 
51 
52 #if	SYM_CONF_GENERIC_SUPPORT
53 /*
54  *  Allocate firmware #1 script area.
55  */
56 #define	SYM_FWA_SCR		sym_fw1a_scr
57 #define	SYM_FWB_SCR		sym_fw1b_scr
58 #define	SYM_FWZ_SCR		sym_fw1z_scr
59 #include "sym_fw1.h"
60 static struct sym_fwa_ofs sym_fw1a_ofs = {
61 	SYM_GEN_FW_A(struct SYM_FWA_SCR)
62 };
63 static struct sym_fwb_ofs sym_fw1b_ofs = {
64 	SYM_GEN_FW_B(struct SYM_FWB_SCR)
65 };
66 static struct sym_fwz_ofs sym_fw1z_ofs = {
67 	SYM_GEN_FW_Z(struct SYM_FWZ_SCR)
68 };
69 #undef	SYM_FWA_SCR
70 #undef	SYM_FWB_SCR
71 #undef	SYM_FWZ_SCR
72 #endif	/* SYM_CONF_GENERIC_SUPPORT */
73 
74 /*
75  *  Allocate firmware #2 script area.
76  */
77 #define	SYM_FWA_SCR		sym_fw2a_scr
78 #define	SYM_FWB_SCR		sym_fw2b_scr
79 #define	SYM_FWZ_SCR		sym_fw2z_scr
80 #include "sym_fw2.h"
81 static struct sym_fwa_ofs sym_fw2a_ofs = {
82 	SYM_GEN_FW_A(struct SYM_FWA_SCR)
83 };
84 static struct sym_fwb_ofs sym_fw2b_ofs = {
85 	SYM_GEN_FW_B(struct SYM_FWB_SCR)
86 	SYM_GEN_B(struct SYM_FWB_SCR, start64)
87 	SYM_GEN_B(struct SYM_FWB_SCR, pm_handle)
88 };
89 static struct sym_fwz_ofs sym_fw2z_ofs = {
90 	SYM_GEN_FW_Z(struct SYM_FWZ_SCR)
91 };
92 #undef	SYM_FWA_SCR
93 #undef	SYM_FWB_SCR
94 #undef	SYM_FWZ_SCR
95 
96 #undef	SYM_GEN_A
97 #undef	SYM_GEN_B
98 #undef	SYM_GEN_Z
99 #undef	PADDR_A
100 #undef	PADDR_B
101 
102 #if	SYM_CONF_GENERIC_SUPPORT
103 /*
104  *  Patch routine for firmware #1.
105  */
106 static void
107 sym_fw1_patch(struct sym_hcb *np)
108 {
109 	struct sym_fw1a_scr *scripta0;
110 	struct sym_fw1b_scr *scriptb0;
111 
112 	scripta0 = (struct sym_fw1a_scr *) np->scripta0;
113 	scriptb0 = (struct sym_fw1b_scr *) np->scriptb0;
114 
115 	/*
116 	 *  Remove LED support if not needed.
117 	 */
118 	if (!(np->features & FE_LED0)) {
119 		scripta0->idle[0]	= cpu_to_scr(SCR_NO_OP);
120 		scripta0->reselected[0]	= cpu_to_scr(SCR_NO_OP);
121 		scripta0->start[0]	= cpu_to_scr(SCR_NO_OP);
122 	}
123 
124 #ifdef SYM_CONF_IARB_SUPPORT
125 	/*
126 	 *    If user does not want to use IMMEDIATE ARBITRATION
127 	 *    when we are reselected while attempting to arbitrate,
128 	 *    patch the SCRIPTS accordingly with a SCRIPT NO_OP.
129 	 */
130 	if (!SYM_CONF_SET_IARB_ON_ARB_LOST)
131 		scripta0->ungetjob[0] = cpu_to_scr(SCR_NO_OP);
132 #endif
133 	/*
134 	 *  Patch some data in SCRIPTS.
135 	 *  - start and done queue initial bus address.
136 	 *  - target bus address table bus address.
137 	 */
138 	scriptb0->startpos[0]	= cpu_to_scr(np->squeue_ba);
139 	scriptb0->done_pos[0]	= cpu_to_scr(np->dqueue_ba);
140 	scriptb0->targtbl[0]	= cpu_to_scr(np->targtbl_ba);
141 }
142 #endif	/* SYM_CONF_GENERIC_SUPPORT */
143 
144 /*
145  *  Patch routine for firmware #2.
146  */
147 static void
148 sym_fw2_patch(struct sym_hcb *np)
149 {
150 	struct sym_fw2a_scr *scripta0;
151 	struct sym_fw2b_scr *scriptb0;
152 
153 	scripta0 = (struct sym_fw2a_scr *) np->scripta0;
154 	scriptb0 = (struct sym_fw2b_scr *) np->scriptb0;
155 
156 	/*
157 	 *  Remove LED support if not needed.
158 	 */
159 	if (!(np->features & FE_LED0)) {
160 		scripta0->idle[0]	= cpu_to_scr(SCR_NO_OP);
161 		scripta0->reselected[0]	= cpu_to_scr(SCR_NO_OP);
162 		scripta0->start[0]	= cpu_to_scr(SCR_NO_OP);
163 	}
164 
165 #if   SYM_CONF_DMA_ADDRESSING_MODE == 2
166 	/*
167 	 *  Remove useless 64 bit DMA specific SCRIPTS,
168 	 *  when this feature is not available.
169 	 */
170 	if (!np->use_dac) {
171 		scripta0->is_dmap_dirty[0] = cpu_to_scr(SCR_NO_OP);
172 		scripta0->is_dmap_dirty[1] = 0;
173 		scripta0->is_dmap_dirty[2] = cpu_to_scr(SCR_NO_OP);
174 		scripta0->is_dmap_dirty[3] = 0;
175 	}
176 #endif
177 
178 #ifdef SYM_CONF_IARB_SUPPORT
179 	/*
180 	 *    If user does not want to use IMMEDIATE ARBITRATION
181 	 *    when we are reselected while attempting to arbitrate,
182 	 *    patch the SCRIPTS accordingly with a SCRIPT NO_OP.
183 	 */
184 	if (!SYM_CONF_SET_IARB_ON_ARB_LOST)
185 		scripta0->ungetjob[0] = cpu_to_scr(SCR_NO_OP);
186 #endif
187 	/*
188 	 *  Patch some variable in SCRIPTS.
189 	 *  - start and done queue initial bus address.
190 	 *  - target bus address table bus address.
191 	 */
192 	scriptb0->startpos[0]	= cpu_to_scr(np->squeue_ba);
193 	scriptb0->done_pos[0]	= cpu_to_scr(np->dqueue_ba);
194 	scriptb0->targtbl[0]	= cpu_to_scr(np->targtbl_ba);
195 
196 	/*
197 	 *  Remove the load of SCNTL4 on reselection if not a C10.
198 	 */
199 	if (!(np->features & FE_C10)) {
200 		scripta0->resel_scntl4[0] = cpu_to_scr(SCR_NO_OP);
201 		scripta0->resel_scntl4[1] = cpu_to_scr(0);
202 	}
203 
204 	/*
205 	 *  Remove a couple of work-arounds specific to C1010 if
206 	 *  they are not desirable. See `sym_fw2.h' for more details.
207 	 */
208 	if (!(np->device_id == PCI_DEVICE_ID_LSI_53C1010_66 &&
209 	      np->revision_id < 0x1 &&
210 	      np->pciclk_khz < 60000)) {
211 		scripta0->datao_phase[0] = cpu_to_scr(SCR_NO_OP);
212 		scripta0->datao_phase[1] = cpu_to_scr(0);
213 	}
214 	if (!(np->device_id == PCI_DEVICE_ID_LSI_53C1010_33 &&
215 	      /* np->revision_id < 0xff */ 1)) {
216 		scripta0->sel_done[0] = cpu_to_scr(SCR_NO_OP);
217 		scripta0->sel_done[1] = cpu_to_scr(0);
218 	}
219 
220 	/*
221 	 *  Patch some other variables in SCRIPTS.
222 	 *  These ones are loaded by the SCRIPTS processor.
223 	 */
224 	scriptb0->pm0_data_addr[0] =
225 		cpu_to_scr(np->scripta_ba +
226 			   offsetof(struct sym_fw2a_scr, pm0_data));
227 	scriptb0->pm1_data_addr[0] =
228 		cpu_to_scr(np->scripta_ba +
229 			   offsetof(struct sym_fw2a_scr, pm1_data));
230 }
231 
232 /*
233  *  Fill the data area in scripts.
234  *  To be done for all firmwares.
235  */
236 static void
237 sym_fw_fill_data (u32 *in, u32 *out)
238 {
239 	int	i;
240 
241 	for (i = 0; i < SYM_CONF_MAX_SG; i++) {
242 		*in++  = SCR_CHMOV_TBL ^ SCR_DATA_IN;
243 		*in++  = offsetof (struct sym_dsb, data[i]);
244 		*out++ = SCR_CHMOV_TBL ^ SCR_DATA_OUT;
245 		*out++ = offsetof (struct sym_dsb, data[i]);
246 	}
247 }
248 
249 /*
250  *  Setup useful script bus addresses.
251  *  To be done for all firmwares.
252  */
253 static void
254 sym_fw_setup_bus_addresses(struct sym_hcb *np, struct sym_fw *fw)
255 {
256 	u32 *pa;
257 	u_short *po;
258 	int i;
259 
260 	/*
261 	 *  Build the bus address table for script A
262 	 *  from the script A offset table.
263 	 */
264 	po = (u_short *) fw->a_ofs;
265 	pa = (u32 *) &np->fwa_bas;
266 	for (i = 0 ; i < sizeof(np->fwa_bas)/sizeof(u32) ; i++)
267 		pa[i] = np->scripta_ba + po[i];
268 
269 	/*
270 	 *  Same for script B.
271 	 */
272 	po = (u_short *) fw->b_ofs;
273 	pa = (u32 *) &np->fwb_bas;
274 	for (i = 0 ; i < sizeof(np->fwb_bas)/sizeof(u32) ; i++)
275 		pa[i] = np->scriptb_ba + po[i];
276 
277 	/*
278 	 *  Same for script Z.
279 	 */
280 	po = (u_short *) fw->z_ofs;
281 	pa = (u32 *) &np->fwz_bas;
282 	for (i = 0 ; i < sizeof(np->fwz_bas)/sizeof(u32) ; i++)
283 		pa[i] = np->scriptz_ba + po[i];
284 }
285 
286 #if	SYM_CONF_GENERIC_SUPPORT
287 /*
288  *  Setup routine for firmware #1.
289  */
290 static void
291 sym_fw1_setup(struct sym_hcb *np, struct sym_fw *fw)
292 {
293 	struct sym_fw1a_scr *scripta0;
294 	struct sym_fw1b_scr *scriptb0;
295 
296 	scripta0 = (struct sym_fw1a_scr *) np->scripta0;
297 	scriptb0 = (struct sym_fw1b_scr *) np->scriptb0;
298 
299 	/*
300 	 *  Fill variable parts in scripts.
301 	 */
302 	sym_fw_fill_data(scripta0->data_in, scripta0->data_out);
303 
304 	/*
305 	 *  Setup bus addresses used from the C code..
306 	 */
307 	sym_fw_setup_bus_addresses(np, fw);
308 }
309 #endif	/* SYM_CONF_GENERIC_SUPPORT */
310 
311 /*
312  *  Setup routine for firmware #2.
313  */
314 static void
315 sym_fw2_setup(struct sym_hcb *np, struct sym_fw *fw)
316 {
317 	struct sym_fw2a_scr *scripta0;
318 	struct sym_fw2b_scr *scriptb0;
319 
320 	scripta0 = (struct sym_fw2a_scr *) np->scripta0;
321 	scriptb0 = (struct sym_fw2b_scr *) np->scriptb0;
322 
323 	/*
324 	 *  Fill variable parts in scripts.
325 	 */
326 	sym_fw_fill_data(scripta0->data_in, scripta0->data_out);
327 
328 	/*
329 	 *  Setup bus addresses used from the C code..
330 	 */
331 	sym_fw_setup_bus_addresses(np, fw);
332 }
333 
334 /*
335  *  Allocate firmware descriptors.
336  */
337 #if	SYM_CONF_GENERIC_SUPPORT
338 static struct sym_fw sym_fw1 = SYM_FW_ENTRY(sym_fw1, "NCR-generic");
339 #endif	/* SYM_CONF_GENERIC_SUPPORT */
340 static struct sym_fw sym_fw2 = SYM_FW_ENTRY(sym_fw2, "LOAD/STORE-based");
341 
342 /*
343  *  Find the most appropriate firmware for a chip.
344  */
345 struct sym_fw *
346 sym_find_firmware(struct sym_chip *chip)
347 {
348 	if (chip->features & FE_LDSTR)
349 		return &sym_fw2;
350 #if	SYM_CONF_GENERIC_SUPPORT
351 	else if (!(chip->features & (FE_PFEN|FE_NOPM|FE_DAC)))
352 		return &sym_fw1;
353 #endif
354 	else
355 		return NULL;
356 }
357 
358 /*
359  *  Bind a script to physical addresses.
360  */
361 void sym_fw_bind_script(struct sym_hcb *np, u32 *start, int len)
362 {
363 	u32 opcode, new, old, tmp1, tmp2;
364 	u32 *end, *cur;
365 	int relocs;
366 
367 	cur = start;
368 	end = start + len/4;
369 
370 	while (cur < end) {
371 
372 		opcode = *cur;
373 
374 		/*
375 		 *  If we forget to change the length
376 		 *  in scripts, a field will be
377 		 *  padded with 0. This is an illegal
378 		 *  command.
379 		 */
380 		if (opcode == 0) {
381 			printf ("%s: ERROR0 IN SCRIPT at %d.\n",
382 				sym_name(np), (int) (cur-start));
383 			++cur;
384 			continue;
385 		};
386 
387 		/*
388 		 *  We use the bogus value 0xf00ff00f ;-)
389 		 *  to reserve data area in SCRIPTS.
390 		 */
391 		if (opcode == SCR_DATA_ZERO) {
392 			*cur++ = 0;
393 			continue;
394 		}
395 
396 		if (DEBUG_FLAGS & DEBUG_SCRIPT)
397 			printf ("%d:  <%x>\n", (int) (cur-start),
398 				(unsigned)opcode);
399 
400 		/*
401 		 *  We don't have to decode ALL commands
402 		 */
403 		switch (opcode >> 28) {
404 		case 0xf:
405 			/*
406 			 *  LOAD / STORE DSA relative, don't relocate.
407 			 */
408 			relocs = 0;
409 			break;
410 		case 0xe:
411 			/*
412 			 *  LOAD / STORE absolute.
413 			 */
414 			relocs = 1;
415 			break;
416 		case 0xc:
417 			/*
418 			 *  COPY has TWO arguments.
419 			 */
420 			relocs = 2;
421 			tmp1 = cur[1];
422 			tmp2 = cur[2];
423 			if ((tmp1 ^ tmp2) & 3) {
424 				printf ("%s: ERROR1 IN SCRIPT at %d.\n",
425 					sym_name(np), (int) (cur-start));
426 			}
427 			/*
428 			 *  If PREFETCH feature not enabled, remove
429 			 *  the NO FLUSH bit if present.
430 			 */
431 			if ((opcode & SCR_NO_FLUSH) &&
432 			    !(np->features & FE_PFEN)) {
433 				opcode = (opcode & ~SCR_NO_FLUSH);
434 			}
435 			break;
436 		case 0x0:
437 			/*
438 			 *  MOVE/CHMOV (absolute address)
439 			 */
440 			if (!(np->features & FE_WIDE))
441 				opcode = (opcode | OPC_MOVE);
442 			relocs = 1;
443 			break;
444 		case 0x1:
445 			/*
446 			 *  MOVE/CHMOV (table indirect)
447 			 */
448 			if (!(np->features & FE_WIDE))
449 				opcode = (opcode | OPC_MOVE);
450 			relocs = 0;
451 			break;
452 #ifdef SYM_CONF_TARGET_ROLE_SUPPORT
453 		case 0x2:
454 			/*
455 			 *  MOVE/CHMOV in target role (absolute address)
456 			 */
457 			opcode &= ~0x20000000;
458 			if (!(np->features & FE_WIDE))
459 				opcode = (opcode & ~OPC_TCHMOVE);
460 			relocs = 1;
461 			break;
462 		case 0x3:
463 			/*
464 			 *  MOVE/CHMOV in target role (table indirect)
465 			 */
466 			opcode &= ~0x20000000;
467 			if (!(np->features & FE_WIDE))
468 				opcode = (opcode & ~OPC_TCHMOVE);
469 			relocs = 0;
470 			break;
471 #endif
472 		case 0x8:
473 			/*
474 			 *  JUMP / CALL
475 			 *  don't relocate if relative :-)
476 			 */
477 			if (opcode & 0x00800000)
478 				relocs = 0;
479 			else if ((opcode & 0xf8400000) == 0x80400000)/*JUMP64*/
480 				relocs = 2;
481 			else
482 				relocs = 1;
483 			break;
484 		case 0x4:
485 		case 0x5:
486 		case 0x6:
487 		case 0x7:
488 			relocs = 1;
489 			break;
490 		default:
491 			relocs = 0;
492 			break;
493 		};
494 
495 		/*
496 		 *  Scriptify:) the opcode.
497 		 */
498 		*cur++ = cpu_to_scr(opcode);
499 
500 		/*
501 		 *  If no relocation, assume 1 argument
502 		 *  and just scriptize:) it.
503 		 */
504 		if (!relocs) {
505 			*cur = cpu_to_scr(*cur);
506 			++cur;
507 			continue;
508 		}
509 
510 		/*
511 		 *  Otherwise performs all needed relocations.
512 		 */
513 		while (relocs--) {
514 			old = *cur;
515 
516 			switch (old & RELOC_MASK) {
517 			case RELOC_REGISTER:
518 				new = (old & ~RELOC_MASK) + np->mmio_ba;
519 				break;
520 			case RELOC_LABEL_A:
521 				new = (old & ~RELOC_MASK) + np->scripta_ba;
522 				break;
523 			case RELOC_LABEL_B:
524 				new = (old & ~RELOC_MASK) + np->scriptb_ba;
525 				break;
526 			case RELOC_SOFTC:
527 				new = (old & ~RELOC_MASK) + np->hcb_ba;
528 				break;
529 			case 0:
530 				/*
531 				 *  Don't relocate a 0 address.
532 				 *  They are mostly used for patched or
533 				 *  script self-modified areas.
534 				 */
535 				if (old == 0) {
536 					new = old;
537 					break;
538 				}
539 				/* fall through */
540 			default:
541 				new = 0;
542 				panic("sym_fw_bind_script: "
543 				      "weird relocation %x\n", old);
544 				break;
545 			}
546 
547 			*cur++ = cpu_to_scr(new);
548 		}
549 	};
550 }
551