Annotation of sys/arch/sparc/dev/zs_kgdb.c, Revision 1.1.1.1
1.1 nbrk 1: /* $OpenBSD: zs_kgdb.c,v 1.2 2005/11/11 15:21:59 fgsch Exp $ */
2: /* $NetBSD: zs_kgdb.c,v 1.1 1997/10/18 00:00:51 gwr Exp $ */
3:
4: /*-
5: * Copyright (c) 1996 The NetBSD Foundation, Inc.
6: * All rights reserved.
7: *
8: * This code is derived from software contributed to The NetBSD Foundation
9: * by Gordon W. Ross.
10: *
11: * Redistribution and use in source and binary forms, with or without
12: * modification, are permitted provided that the following conditions
13: * are met:
14: * 1. Redistributions of source code must retain the above copyright
15: * notice, this list of conditions and the following disclaimer.
16: * 2. Redistributions in binary form must reproduce the above copyright
17: * notice, this list of conditions and the following disclaimer in the
18: * documentation and/or other materials provided with the distribution.
19: * 3. All advertising materials mentioning features or use of this software
20: * must display the following acknowledgement:
21: * This product includes software developed by the NetBSD
22: * Foundation, Inc. and its contributors.
23: * 4. Neither the name of The NetBSD Foundation nor the names of its
24: * contributors may be used to endorse or promote products derived
25: * from this software without specific prior written permission.
26: *
27: * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
28: * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
29: * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
30: * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
31: * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
32: * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
33: * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
34: * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
35: * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
36: * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
37: * POSSIBILITY OF SUCH DAMAGE.
38: */
39:
40: /*
41: * Hooks for kgdb when attached via the z8530 driver
42: *
43: * To use this, build a kernel with: option KGDB, and
44: * boot that kernel with "-d". (The kernel will call
45: * zs_kgdb_init, kgdb_connect.) When the console prints
46: * "kgdb waiting..." you run "gdb -k kernel" and do:
47: * (gdb) set remotebaud 19200
48: * (gdb) target remote /dev/ttyb
49: */
50:
51: #include <sys/param.h>
52: #include <sys/systm.h>
53: #include <sys/proc.h>
54: #include <sys/device.h>
55: #include <sys/conf.h>
56: #include <sys/ioctl.h>
57: #include <sys/kernel.h>
58: #include <sys/syslog.h>
59: #include <sys/kgdb.h>
60:
61: #include <sparc/dev/z8530reg.h>
62: #include <machine/z8530var.h>
63: #include <sparc/dev/cons.h>
64:
65: /* The Sun3 provides a 4.9152 MHz clock to the ZS chips. */
66: #define PCLK (9600 * 512) /* PCLK pin input clock rate */
67: #define ZSHARD_PRI 6 /* Wired on the CPU board... */
68:
69: #define ZS_DELAY() (CPU_ISSUN4C ? (0) : delay(2))
70:
71: /* The layout of this is hardware-dependent (padding, order). */
72: struct zschan {
73: volatile u_char zc_csr; /* ctrl,status, and indirect access */
74: u_char zc_xxx0;
75: volatile u_char zc_data; /* data */
76: u_char zc_xxx1;
77: };
78:
79: void zs_setparam(struct zs_chanstate *, int, int);
80: struct zsops zsops_kgdb;
81:
82: u_char zs_kgdb_regs[16] = {
83: 0, /* 0: CMD (reset, etc.) */
84: 0, /* 1: ~(ZSWR1_RIE | ZSWR1_TIE | ZSWR1_SIE) */
85: 0, /* 2: IVECT */
86: ZSWR3_RX_8 | ZSWR3_RX_ENABLE,
87: ZSWR4_CLK_X16 | ZSWR4_ONESB | ZSWR4_EVENP,
88: ZSWR5_TX_8 | ZSWR5_TX_ENABLE,
89: 0, /* 6: TXSYNC/SYNCLO */
90: 0, /* 7: RXSYNC/SYNCHI */
91: 0, /* 8: alias for data port */
92: ZSWR9_MASTER_IE | ZSWR9_NO_VECTOR,
93: 0, /*10: Misc. TX/RX control bits */
94: ZSWR11_TXCLK_BAUD | ZSWR11_RXCLK_BAUD,
95: 14, /*12: BAUDLO (default=9600) */
96: 0, /*13: BAUDHI (default=9600) */
97: ZSWR14_BAUD_ENA | ZSWR14_BAUD_FROM_PCLK,
98: ZSWR15_BREAK_IE | ZSWR15_DCD_IE,
99: };
100:
101: /*
102: * This replaces "zs_reset()" in the sparc driver.
103: */
104: void
105: zs_setparam(cs, iena, rate)
106: struct zs_chanstate *cs;
107: int iena;
108: int rate;
109: {
110: int s, tconst;
111:
112: bcopy(zs_kgdb_regs, cs->cs_preg, 16);
113:
114: if (iena) {
115: cs->cs_preg[1] = ZSWR1_RIE | ZSWR1_SIE;
116: }
117:
118: /* Initialize the speed, etc. */
119: tconst = BPS_TO_TCONST(cs->cs_brg_clk, rate);
120: cs->cs_preg[5] |= ZSWR5_DTR | ZSWR5_RTS;
121: cs->cs_preg[12] = tconst;
122: cs->cs_preg[13] = tconst >> 8;
123:
124: s = splhigh();
125: zs_loadchannelregs(cs);
126: splx(s);
127: }
128:
129: /*
130: * Set up for kgdb; called at boot time before configuration.
131: * KGDB interrupts will be enabled later when zs0 is configured.
132: * Called after cninit(), so printf() etc. works.
133: */
134: void
135: zs_kgdb_init()
136: {
137: struct zs_chanstate cs;
138: volatile struct zschan *zc;
139: int channel, zsc_unit;
140:
141: /* printf("zs_kgdb_init: kgdb_dev=0x%x\n", kgdb_dev); */
142: if (major(kgdb_dev) != zs_major)
143: return;
144:
145: /* Note: (ttya,ttyb) on zsc1, and (ttyc,ttyd) on zsc0 */
146: zsc_unit = (kgdb_dev & 2) ? 0 : 1;
147: channel = kgdb_dev & 1;
148: printf("zs_kgdb_init: attaching tty%c at %d baud\n",
149: 'a' + (kgdb_dev & 3), kgdb_rate);
150:
151: /* Setup temporary chanstate. */
152: bzero((caddr_t)&cs, sizeof(cs));
153: zc = zs_get_chan_addr(zsc_unit, channel);
154: if (zc == NULL) {
155: printf("zs_kgdb_init: zs not mapped.\n");
156: kgdb_dev = -1;
157: return;
158: }
159:
160: cs.cs_channel = channel;
161: cs.cs_brg_clk = PCLK / 16;
162: cs.cs_reg_csr = &zc->zc_csr;
163: cs.cs_reg_data = &zc->zc_data;
164:
165: /* Now set parameters. (interrupts disabled) */
166: zs_setparam(&cs, 0, kgdb_rate);
167:
168: /* Store the getc/putc functions and arg. */
169: kgdb_attach(zs_getc, zs_putc, (void *)zc);
170: }
171:
172: /*
173: * This is a "hook" called by zstty_attach to allow the tty
174: * to be "taken over" for exclusive use by kgdb.
175: * Return non-zero if this is the kgdb port.
176: *
177: * Set the speed to kgdb_rate, CS8, etc.
178: */
179: int
180: zs_check_kgdb(cs, dev)
181: struct zs_chanstate *cs;
182: int dev;
183: {
184:
185: if (dev != kgdb_dev)
186: return (0);
187:
188: /*
189: * Yes, this is port in use by kgdb.
190: */
191: cs->cs_private = NULL;
192: cs->cs_ops = &zsops_kgdb;
193:
194: /* Now set parameters. (interrupts enabled) */
195: zs_setparam(cs, 1, kgdb_rate);
196:
197: return (1);
198: }
199:
200: /*
201: * KGDB framing character received: enter kernel debugger. This probably
202: * should time out after a few seconds to avoid hanging on spurious input.
203: */
204: void
205: zskgdb(cs)
206: struct zs_chanstate *cs;
207: {
208: int unit = minor(kgdb_dev);
209:
210: printf("zstty%d: kgdb interrupt\n", unit);
211: /* This will trap into the debugger. */
212: kgdb_connect(1);
213: }
214:
215:
216: /****************************************************************
217: * Interface to the lower layer (zscc)
218: ****************************************************************/
219:
220: void zs_kgdb_rxint(struct zs_chanstate *);
221: void zs_kgdb_txint(struct zs_chanstate *);
222: void zs_kgdb_stint(struct zs_chanstate *, int);
223: void zs_kgdb_softint(struct zs_chanstate *);
224:
225: int kgdb_input_lost;
226:
227: void
228: zs_kgdb_rxint(cs)
229: struct zs_chanstate *cs;
230: {
231: register u_char c, rr1;
232:
233: /*
234: * First read the status, because reading the received char
235: * destroys the status of this char.
236: */
237: rr1 = zs_read_reg(cs, 1);
238: c = zs_read_data(cs);
239:
240: if (rr1 & (ZSRR1_FE | ZSRR1_DO | ZSRR1_PE)) {
241: /* Clear the receive error. */
242: zs_write_csr(cs, ZSWR0_RESET_ERRORS);
243: }
244:
245: if (c == KGDB_START) {
246: zskgdb(cs);
247: } else {
248: kgdb_input_lost++;
249: }
250: }
251:
252: void
253: zs_kgdb_txint(cs)
254: register struct zs_chanstate *cs;
255: {
256: register int rr0;
257:
258: rr0 = zs_read_csr(cs);
259: zs_write_csr(cs, ZSWR0_RESET_TXINT);
260: }
261:
262: void
263: zs_kgdb_stint(cs, force)
264: register struct zs_chanstate *cs;
265: int force;
266: {
267: register int rr0;
268:
269: rr0 = zs_read_csr(cs);
270: zs_write_csr(cs, ZSWR0_RESET_STATUS);
271:
272: /*
273: * Check here for console break, so that we can abort
274: * even when interrupts are locking up the machine.
275: */
276: if (rr0 & ZSRR0_BREAK) {
277: zskgdb(cs);
278: }
279: }
280:
281: void
282: zs_kgdb_softint(cs)
283: struct zs_chanstate *cs;
284: {
285: printf("zs_kgdb_softint?\n");
286: }
287:
288: struct zsops zsops_kgdb = {
289: zs_kgdb_rxint, /* receive char available */
290: zs_kgdb_stint, /* external/status */
291: zs_kgdb_txint, /* xmit buffer empty */
292: zs_kgdb_softint, /* process software interrupt */
293: };
CVSweb