dcons_os.c revision 136467
1136467Ssimokawa/*
2136467Ssimokawa * Copyright (C) 2003,2004
3136467Ssimokawa * 	Hidetoshi Shimokawa. All rights reserved.
4136467Ssimokawa *
5136467Ssimokawa * Redistribution and use in source and binary forms, with or without
6136467Ssimokawa * modification, are permitted provided that the following conditions
7136467Ssimokawa * are met:
8136467Ssimokawa * 1. Redistributions of source code must retain the above copyright
9136467Ssimokawa *    notice, this list of conditions and the following disclaimer.
10136467Ssimokawa * 2. Redistributions in binary form must reproduce the above copyright
11136467Ssimokawa *    notice, this list of conditions and the following disclaimer in the
12136467Ssimokawa *    documentation and/or other materials provided with the distribution.
13136467Ssimokawa * 3. All advertising materials mentioning features or use of this software
14136467Ssimokawa *    must display the following acknowledgement:
15136467Ssimokawa *
16136467Ssimokawa *	This product includes software developed by Hidetoshi Shimokawa.
17136467Ssimokawa *
18136467Ssimokawa * 4. Neither the name of the author nor the names of its contributors
19136467Ssimokawa *    may be used to endorse or promote products derived from this software
20136467Ssimokawa *    without specific prior written permission.
21136467Ssimokawa *
22136467Ssimokawa * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23136467Ssimokawa * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24136467Ssimokawa * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25136467Ssimokawa * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26136467Ssimokawa * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27136467Ssimokawa * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28136467Ssimokawa * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29136467Ssimokawa * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30136467Ssimokawa * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31136467Ssimokawa * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32136467Ssimokawa * SUCH DAMAGE.
33136467Ssimokawa *
34136467Ssimokawa * $FreeBSD: head/sys/dev/dcons/dcons_os.c 136467 2004-10-13 05:38:42Z simokawa $
35136467Ssimokawa */
36136467Ssimokawa
37136467Ssimokawa#include <sys/param.h>
38136467Ssimokawa#if __FreeBSD_version >= 502122
39136467Ssimokawa#include <sys/kdb.h>
40136467Ssimokawa#include <gdb/gdb.h>
41136467Ssimokawa#endif
42136467Ssimokawa#include <sys/kernel.h>
43136467Ssimokawa#include <sys/module.h>
44136467Ssimokawa#include <sys/systm.h>
45136467Ssimokawa#include <sys/types.h>
46136467Ssimokawa#include <sys/conf.h>
47136467Ssimokawa#include <sys/cons.h>
48136467Ssimokawa#include <sys/consio.h>
49136467Ssimokawa#include <sys/tty.h>
50136467Ssimokawa#include <sys/malloc.h>
51136467Ssimokawa#include <sys/proc.h>
52136467Ssimokawa#include <sys/ucred.h>
53136467Ssimokawa
54136467Ssimokawa#include <machine/bus.h>
55136467Ssimokawa
56136467Ssimokawa#ifdef __DragonFly__
57136467Ssimokawa#include "dcons.h"
58136467Ssimokawa#include "dcons_os.h"
59136467Ssimokawa#else
60136467Ssimokawa#include <dev/dcons/dcons.h>
61136467Ssimokawa#include <dev/dcons/dcons_os.h>
62136467Ssimokawa#endif
63136467Ssimokawa
64136467Ssimokawa#include <ddb/ddb.h>
65136467Ssimokawa#include <sys/reboot.h>
66136467Ssimokawa
67136467Ssimokawa#include <sys/sysctl.h>
68136467Ssimokawa
69136467Ssimokawa#include <vm/vm.h>
70136467Ssimokawa#include <vm/vm_param.h>
71136467Ssimokawa#include <vm/pmap.h>
72136467Ssimokawa
73136467Ssimokawa#include "opt_ddb.h"
74136467Ssimokawa#include "opt_comconsole.h"
75136467Ssimokawa#include "opt_dcons.h"
76136467Ssimokawa
77136467Ssimokawa#ifndef DCONS_POLL_HZ
78136467Ssimokawa#define DCONS_POLL_HZ	100
79136467Ssimokawa#endif
80136467Ssimokawa
81136467Ssimokawa#ifndef DCONS_BUF_SIZE
82136467Ssimokawa#define DCONS_BUF_SIZE (16*1024)
83136467Ssimokawa#endif
84136467Ssimokawa
85136467Ssimokawa#ifndef DCONS_FORCE_CONSOLE
86136467Ssimokawa#define DCONS_FORCE_CONSOLE	0	/* Mostly for FreeBSD-4/DragonFly */
87136467Ssimokawa#endif
88136467Ssimokawa
89136467Ssimokawa#ifndef DCONS_FORCE_GDB
90136467Ssimokawa#define DCONS_FORCE_GDB	1
91136467Ssimokawa#endif
92136467Ssimokawa
93136467Ssimokawa#if __FreeBSD_version >= 500101
94136467Ssimokawa#define CONS_NODEV	1
95136467Ssimokawa#if __FreeBSD_version < 502122
96136467Ssimokawastatic struct consdev gdbconsdev;
97136467Ssimokawa#endif
98136467Ssimokawa#endif
99136467Ssimokawa
100136467Ssimokawastatic d_open_t		dcons_open;
101136467Ssimokawastatic d_close_t	dcons_close;
102136467Ssimokawa#if defined(__DragonFly__) || __FreeBSD_version < 500104
103136467Ssimokawastatic d_ioctl_t	dcons_ioctl;
104136467Ssimokawa#endif
105136467Ssimokawa
106136467Ssimokawastatic struct cdevsw dcons_cdevsw = {
107136467Ssimokawa#ifdef __DragonFly__
108136467Ssimokawa#define CDEV_MAJOR      184
109136467Ssimokawa	"dcons", CDEV_MAJOR, D_TTY, NULL, 0,
110136467Ssimokawa	dcons_open, dcons_close, ttyread, ttywrite, dcons_ioctl,
111136467Ssimokawa	ttypoll, nommap, nostrategy, nodump, nopsize,
112136467Ssimokawa#elif __FreeBSD_version >= 500104
113136467Ssimokawa	.d_version =	D_VERSION,
114136467Ssimokawa	.d_open =	dcons_open,
115136467Ssimokawa	.d_close =	dcons_close,
116136467Ssimokawa	.d_name =	"dcons",
117136467Ssimokawa	.d_flags =	D_TTY | D_NEEDGIANT,
118136467Ssimokawa#else
119136467Ssimokawa#define CDEV_MAJOR      184
120136467Ssimokawa	/* open */	dcons_open,
121136467Ssimokawa	/* close */	dcons_close,
122136467Ssimokawa	/* read */	ttyread,
123136467Ssimokawa	/* write */	ttywrite,
124136467Ssimokawa	/* ioctl */	dcons_ioctl,
125136467Ssimokawa	/* poll */	ttypoll,
126136467Ssimokawa	/* mmap */	nommap,
127136467Ssimokawa	/* strategy */	nostrategy,
128136467Ssimokawa	/* name */	"dcons",
129136467Ssimokawa	/* major */	CDEV_MAJOR,
130136467Ssimokawa	/* dump */	nodump,
131136467Ssimokawa	/* psize */	nopsize,
132136467Ssimokawa	/* flags */	D_TTY,
133136467Ssimokawa#endif
134136467Ssimokawa};
135136467Ssimokawa
136136467Ssimokawa#ifndef KLD_MODULE
137136467Ssimokawastatic char bssbuf[DCONS_BUF_SIZE];	/* buf in bss */
138136467Ssimokawa#endif
139136467Ssimokawa
140136467Ssimokawa/* global data */
141136467Ssimokawastatic struct dcons_global dg;
142136467Ssimokawastruct dcons_global *dcons_conf;
143136467Ssimokawastatic int poll_hz = DCONS_POLL_HZ;
144136467Ssimokawa
145136467Ssimokawastatic struct dcons_softc sc[DCONS_NPORT];
146136467Ssimokawa
147136467SsimokawaSYSCTL_NODE(_kern, OID_AUTO, dcons, CTLFLAG_RD, 0, "Dumb Console");
148136467SsimokawaSYSCTL_INT(_kern_dcons, OID_AUTO, poll_hz, CTLFLAG_RW, &poll_hz, 0,
149136467Ssimokawa				"dcons polling rate");
150136467Ssimokawa
151136467Ssimokawastatic int drv_init = 0;
152136467Ssimokawastatic struct callout dcons_callout;
153136467Ssimokawastruct dcons_buf *dcons_buf;		/* for local dconschat */
154136467Ssimokawa
155136467Ssimokawa#ifdef __DragonFly__
156136467Ssimokawa#define DEV	dev_t
157136467Ssimokawa#define THREAD	d_thread_t
158136467Ssimokawa#elif __FreeBSD_version < 500000
159136467Ssimokawa#define DEV	dev_t
160136467Ssimokawa#define THREAD	struct proc
161136467Ssimokawa#else
162136467Ssimokawa#define DEV	struct cdev *
163136467Ssimokawa#define THREAD	struct thread
164136467Ssimokawa#endif
165136467Ssimokawa
166136467Ssimokawa
167136467Ssimokawastatic void	dcons_tty_start(struct tty *);
168136467Ssimokawastatic int	dcons_tty_param(struct tty *, struct termios *);
169136467Ssimokawastatic void	dcons_timeout(void *);
170136467Ssimokawastatic int	dcons_drv_init(int);
171136467Ssimokawa
172136467Ssimokawastatic cn_probe_t	dcons_cnprobe;
173136467Ssimokawastatic cn_init_t	dcons_cninit;
174136467Ssimokawastatic cn_getc_t	dcons_cngetc;
175136467Ssimokawastatic cn_checkc_t 	dcons_cncheckc;
176136467Ssimokawastatic cn_putc_t	dcons_cnputc;
177136467Ssimokawa
178136467SsimokawaCONS_DRIVER(dcons, dcons_cnprobe, dcons_cninit, NULL, dcons_cngetc,
179136467Ssimokawa    dcons_cncheckc, dcons_cnputc, NULL);
180136467Ssimokawa
181136467Ssimokawa#if __FreeBSD_version >= 502122
182136467Ssimokawastatic gdb_probe_f dcons_dbg_probe;
183136467Ssimokawastatic gdb_init_f dcons_dbg_init;
184136467Ssimokawastatic gdb_term_f dcons_dbg_term;
185136467Ssimokawastatic gdb_getc_f dcons_dbg_getc;
186136467Ssimokawastatic gdb_checkc_f dcons_dbg_checkc;
187136467Ssimokawastatic gdb_putc_f dcons_dbg_putc;
188136467Ssimokawa
189136467SsimokawaGDB_DBGPORT(dcons, dcons_dbg_probe, dcons_dbg_init, dcons_dbg_term,
190136467Ssimokawa    dcons_dbg_checkc, dcons_dbg_getc, dcons_dbg_putc);
191136467Ssimokawa
192136467Ssimokawaextern struct gdb_dbgport *gdb_cur;
193136467Ssimokawa#endif
194136467Ssimokawa
195136467Ssimokawa#if (KDB || DDB) && ALT_BREAK_TO_DEBUGGER
196136467Ssimokawastatic int
197136467Ssimokawadcons_check_break(struct dcons_softc *dc, int c)
198136467Ssimokawa{
199136467Ssimokawa	if (c < 0)
200136467Ssimokawa		return (c);
201136467Ssimokawa
202136467Ssimokawa#if __FreeBSD_version >= 502122
203136467Ssimokawa	if (kdb_alt_break(c, &dc->brk_state)) {
204136467Ssimokawa		if ((dc->flags & DC_GDB) != 0) {
205136467Ssimokawa			if (gdb_cur == &dcons_gdb_dbgport) {
206136467Ssimokawa				kdb_dbbe_select("gdb");
207136467Ssimokawa				breakpoint();
208136467Ssimokawa			}
209136467Ssimokawa		} else
210136467Ssimokawa			breakpoint();
211136467Ssimokawa	}
212136467Ssimokawa#else
213136467Ssimokawa	switch (dc->brk_state) {
214136467Ssimokawa	case STATE1:
215136467Ssimokawa		if (c == KEY_TILDE)
216136467Ssimokawa			dc->brk_state = STATE2;
217136467Ssimokawa		else
218136467Ssimokawa			dc->brk_state = STATE0;
219136467Ssimokawa		break;
220136467Ssimokawa	case STATE2:
221136467Ssimokawa		dc->brk_state = STATE0;
222136467Ssimokawa		if (c == KEY_CTRLB) {
223136467Ssimokawa#if DCONS_FORCE_GDB
224136467Ssimokawa			if (dc->flags & DC_GDB)
225136467Ssimokawa				boothowto |= RB_GDB;
226136467Ssimokawa#endif
227136467Ssimokawa			breakpoint();
228136467Ssimokawa		}
229136467Ssimokawa	}
230136467Ssimokawa	if (c == KEY_CR)
231136467Ssimokawa		dc->brk_state = STATE1;
232136467Ssimokawa#endif
233136467Ssimokawa	return (c);
234136467Ssimokawa}
235136467Ssimokawa#else
236136467Ssimokawa#define	dcons_check_break(dc, c)	(c)
237136467Ssimokawa#endif
238136467Ssimokawa
239136467Ssimokawastatic int
240136467Ssimokawadcons_os_checkc(struct dcons_softc *dc)
241136467Ssimokawa{
242136467Ssimokawa	int c;
243136467Ssimokawa
244136467Ssimokawa	if (dg.dma_tag != NULL)
245136467Ssimokawa		bus_dmamap_sync(dg.dma_tag, dg.dma_map, BUS_DMASYNC_POSTREAD);
246136467Ssimokawa
247136467Ssimokawa	c = dcons_check_break(dc, dcons_checkc(dc));
248136467Ssimokawa
249136467Ssimokawa	if (dg.dma_tag != NULL)
250136467Ssimokawa		bus_dmamap_sync(dg.dma_tag, dg.dma_map, BUS_DMASYNC_PREREAD);
251136467Ssimokawa
252136467Ssimokawa	return (c);
253136467Ssimokawa}
254136467Ssimokawa
255136467Ssimokawastatic int
256136467Ssimokawadcons_os_getc(struct dcons_softc *dc)
257136467Ssimokawa{
258136467Ssimokawa	int c;
259136467Ssimokawa
260136467Ssimokawa	while ((c = dcons_os_checkc(dc)) == -1);
261136467Ssimokawa
262136467Ssimokawa	return (c & 0xff);
263136467Ssimokawa}
264136467Ssimokawa
265136467Ssimokawastatic void
266136467Ssimokawadcons_os_putc(struct dcons_softc *dc, int c)
267136467Ssimokawa{
268136467Ssimokawa	if (dg.dma_tag != NULL)
269136467Ssimokawa		bus_dmamap_sync(dg.dma_tag, dg.dma_map, BUS_DMASYNC_POSTWRITE);
270136467Ssimokawa
271136467Ssimokawa	dcons_putc(dc, c);
272136467Ssimokawa
273136467Ssimokawa	if (dg.dma_tag != NULL)
274136467Ssimokawa		bus_dmamap_sync(dg.dma_tag, dg.dma_map, BUS_DMASYNC_PREWRITE);
275136467Ssimokawa}
276136467Ssimokawastatic int
277136467Ssimokawadcons_open(DEV dev, int flag, int mode, THREAD *td)
278136467Ssimokawa{
279136467Ssimokawa	struct tty *tp;
280136467Ssimokawa	int unit, error, s;
281136467Ssimokawa
282136467Ssimokawa	unit = minor(dev);
283136467Ssimokawa	if (unit != 0)
284136467Ssimokawa		return (ENXIO);
285136467Ssimokawa
286136467Ssimokawa	tp = dev->si_tty = ttymalloc(dev->si_tty);
287136467Ssimokawa	tp->t_oproc = dcons_tty_start;
288136467Ssimokawa	tp->t_param = dcons_tty_param;
289136467Ssimokawa	tp->t_stop = nottystop;
290136467Ssimokawa	tp->t_dev = dev;
291136467Ssimokawa
292136467Ssimokawa	error = 0;
293136467Ssimokawa
294136467Ssimokawa	s = spltty();
295136467Ssimokawa	if ((tp->t_state & TS_ISOPEN) == 0) {
296136467Ssimokawa		tp->t_state |= TS_CARR_ON;
297136467Ssimokawa		ttychars(tp);
298136467Ssimokawa		tp->t_iflag = TTYDEF_IFLAG;
299136467Ssimokawa		tp->t_oflag = TTYDEF_OFLAG;
300136467Ssimokawa		tp->t_cflag = TTYDEF_CFLAG|CLOCAL;
301136467Ssimokawa		tp->t_lflag = TTYDEF_LFLAG;
302136467Ssimokawa		tp->t_ispeed = tp->t_ospeed = TTYDEF_SPEED;
303136467Ssimokawa		ttsetwater(tp);
304136467Ssimokawa	} else if ((tp->t_state & TS_XCLUDE) && suser(td)) {
305136467Ssimokawa		splx(s);
306136467Ssimokawa		return (EBUSY);
307136467Ssimokawa	}
308136467Ssimokawa	splx(s);
309136467Ssimokawa
310136467Ssimokawa#if __FreeBSD_version < 502113
311136467Ssimokawa	error = (*linesw[tp->t_line].l_open)(dev, tp);
312136467Ssimokawa#else
313136467Ssimokawa	error = ttyld_open(tp, dev);
314136467Ssimokawa#endif
315136467Ssimokawa
316136467Ssimokawa	return (error);
317136467Ssimokawa}
318136467Ssimokawa
319136467Ssimokawastatic int
320136467Ssimokawadcons_close(DEV dev, int flag, int mode, THREAD *td)
321136467Ssimokawa{
322136467Ssimokawa	int	unit;
323136467Ssimokawa	struct	tty *tp;
324136467Ssimokawa
325136467Ssimokawa	unit = minor(dev);
326136467Ssimokawa	if (unit != 0)
327136467Ssimokawa		return (ENXIO);
328136467Ssimokawa
329136467Ssimokawa	tp = dev->si_tty;
330136467Ssimokawa	if (tp->t_state & TS_ISOPEN) {
331136467Ssimokawa#if __FreeBSD_version < 502113
332136467Ssimokawa		(*linesw[tp->t_line].l_close)(tp, flag);
333136467Ssimokawa		ttyclose(tp);
334136467Ssimokawa#else
335136467Ssimokawa		ttyld_close(tp, flag);
336136467Ssimokawa		tty_close(tp);
337136467Ssimokawa#endif
338136467Ssimokawa	}
339136467Ssimokawa
340136467Ssimokawa	return (0);
341136467Ssimokawa}
342136467Ssimokawa
343136467Ssimokawa#if defined(__DragonFly__) || __FreeBSD_version < 500104
344136467Ssimokawastatic int
345136467Ssimokawadcons_ioctl(DEV dev, u_long cmd, caddr_t data, int flag, THREAD *td)
346136467Ssimokawa{
347136467Ssimokawa	int	unit;
348136467Ssimokawa	struct	tty *tp;
349136467Ssimokawa	int	error;
350136467Ssimokawa
351136467Ssimokawa	unit = minor(dev);
352136467Ssimokawa	if (unit != 0)
353136467Ssimokawa		return (ENXIO);
354136467Ssimokawa
355136467Ssimokawa	tp = dev->si_tty;
356136467Ssimokawa	error = (*linesw[tp->t_line].l_ioctl)(tp, cmd, data, flag, td);
357136467Ssimokawa	if (error != ENOIOCTL)
358136467Ssimokawa		return (error);
359136467Ssimokawa
360136467Ssimokawa	error = ttioctl(tp, cmd, data, flag);
361136467Ssimokawa	if (error != ENOIOCTL)
362136467Ssimokawa		return (error);
363136467Ssimokawa
364136467Ssimokawa	return (ENOTTY);
365136467Ssimokawa}
366136467Ssimokawa#endif
367136467Ssimokawa
368136467Ssimokawastatic int
369136467Ssimokawadcons_tty_param(struct tty *tp, struct termios *t)
370136467Ssimokawa{
371136467Ssimokawa	tp->t_ispeed = t->c_ispeed;
372136467Ssimokawa	tp->t_ospeed = t->c_ospeed;
373136467Ssimokawa	tp->t_cflag = t->c_cflag;
374136467Ssimokawa	return 0;
375136467Ssimokawa}
376136467Ssimokawa
377136467Ssimokawastatic void
378136467Ssimokawadcons_tty_start(struct tty *tp)
379136467Ssimokawa{
380136467Ssimokawa	struct dcons_softc *dc;
381136467Ssimokawa	int s;
382136467Ssimokawa
383136467Ssimokawa	dc = (struct dcons_softc *)tp->t_dev->si_drv1;
384136467Ssimokawa	s = spltty();
385136467Ssimokawa	if (tp->t_state & (TS_TIMEOUT | TS_TTSTOP)) {
386136467Ssimokawa		ttwwakeup(tp);
387136467Ssimokawa		return;
388136467Ssimokawa	}
389136467Ssimokawa
390136467Ssimokawa	tp->t_state |= TS_BUSY;
391136467Ssimokawa	while (tp->t_outq.c_cc != 0)
392136467Ssimokawa		dcons_os_putc(dc, getc(&tp->t_outq));
393136467Ssimokawa	tp->t_state &= ~TS_BUSY;
394136467Ssimokawa
395136467Ssimokawa	ttwwakeup(tp);
396136467Ssimokawa	splx(s);
397136467Ssimokawa}
398136467Ssimokawa
399136467Ssimokawastatic void
400136467Ssimokawadcons_timeout(void *v)
401136467Ssimokawa{
402136467Ssimokawa	struct	tty *tp;
403136467Ssimokawa	struct dcons_softc *dc;
404136467Ssimokawa	int i, c, polltime;
405136467Ssimokawa
406136467Ssimokawa	for (i = 0; i < DCONS_NPORT; i ++) {
407136467Ssimokawa		dc = &sc[i];
408136467Ssimokawa		tp = ((DEV)dc->dev)->si_tty;
409136467Ssimokawa		while ((c = dcons_os_checkc(dc)) != -1)
410136467Ssimokawa			if (tp->t_state & TS_ISOPEN)
411136467Ssimokawa#if __FreeBSD_version < 502113
412136467Ssimokawa				(*linesw[tp->t_line].l_rint)(c, tp);
413136467Ssimokawa#else
414136467Ssimokawa				ttyld_rint(tp, c);
415136467Ssimokawa#endif
416136467Ssimokawa	}
417136467Ssimokawa	polltime = hz / poll_hz;
418136467Ssimokawa	if (polltime < 1)
419136467Ssimokawa		polltime = 1;
420136467Ssimokawa	callout_reset(&dcons_callout, polltime, dcons_timeout, tp);
421136467Ssimokawa}
422136467Ssimokawa
423136467Ssimokawastatic void
424136467Ssimokawadcons_cnprobe(struct consdev *cp)
425136467Ssimokawa{
426136467Ssimokawa#ifdef __DragonFly__
427136467Ssimokawa	cp->cn_dev = make_dev(&dcons_cdevsw, DCONS_CON,
428136467Ssimokawa	    UID_ROOT, GID_WHEEL, 0600, "dcons");
429136467Ssimokawa#elif __FreeBSD_version >= 501109
430136467Ssimokawa	sprintf(cp->cn_name, "dcons");
431136467Ssimokawa#else
432136467Ssimokawa	cp->cn_dev = makedev(CDEV_MAJOR, DCONS_CON);
433136467Ssimokawa#endif
434136467Ssimokawa#if DCONS_FORCE_CONSOLE
435136467Ssimokawa	cp->cn_pri = CN_REMOTE;
436136467Ssimokawa#else
437136467Ssimokawa	cp->cn_pri = CN_NORMAL;
438136467Ssimokawa#endif
439136467Ssimokawa}
440136467Ssimokawa
441136467Ssimokawastatic void
442136467Ssimokawadcons_cninit(struct consdev *cp)
443136467Ssimokawa{
444136467Ssimokawa	dcons_drv_init(0);
445136467Ssimokawa#if CONS_NODEV
446136467Ssimokawa	cp->cn_arg
447136467Ssimokawa#else
448136467Ssimokawa	cp->cn_dev->si_drv1
449136467Ssimokawa#endif
450136467Ssimokawa		= (void *)&sc[DCONS_CON]; /* share port0 with unit0 */
451136467Ssimokawa}
452136467Ssimokawa
453136467Ssimokawa#if CONS_NODEV
454136467Ssimokawastatic int
455136467Ssimokawadcons_cngetc(struct consdev *cp)
456136467Ssimokawa{
457136467Ssimokawa	struct dcons_softc *dc = (struct dcons_softc *)cp->cn_arg;
458136467Ssimokawa	return (dcons_os_getc(dc));
459136467Ssimokawa}
460136467Ssimokawastatic int
461136467Ssimokawadcons_cncheckc(struct consdev *cp)
462136467Ssimokawa{
463136467Ssimokawa	struct dcons_softc *dc = (struct dcons_softc *)cp->cn_arg;
464136467Ssimokawa	return (dcons_os_checkc(dc));
465136467Ssimokawa}
466136467Ssimokawastatic void
467136467Ssimokawadcons_cnputc(struct consdev *cp, int c)
468136467Ssimokawa{
469136467Ssimokawa	struct dcons_softc *dc = (struct dcons_softc *)cp->cn_arg;
470136467Ssimokawa	dcons_os_putc(dc, c);
471136467Ssimokawa}
472136467Ssimokawa#else
473136467Ssimokawastatic int
474136467Ssimokawadcons_cngetc(DEV dev)
475136467Ssimokawa{
476136467Ssimokawa	struct dcons_softc *dc = (struct dcons_softc *)dev->si_drv1;
477136467Ssimokawa	return (dcons_os_getc(dc));
478136467Ssimokawa}
479136467Ssimokawastatic int
480136467Ssimokawadcons_cncheckc(DEV dev)
481136467Ssimokawa{
482136467Ssimokawa	struct dcons_softc *dc = (struct dcons_softc *)dev->si_drv1;
483136467Ssimokawa	return (dcons_os_checkc(dc));
484136467Ssimokawa}
485136467Ssimokawastatic void
486136467Ssimokawadcons_cnputc(DEV dev, int c)
487136467Ssimokawa{
488136467Ssimokawa	struct dcons_softc *dc = (struct dcons_softc *)dev->si_drv1;
489136467Ssimokawa	dcons_os_putc(dc, c);
490136467Ssimokawa}
491136467Ssimokawa#endif
492136467Ssimokawa
493136467Ssimokawastatic int
494136467Ssimokawadcons_drv_init(int stage)
495136467Ssimokawa{
496136467Ssimokawa	int addr, size;
497136467Ssimokawa
498136467Ssimokawa	if (drv_init)
499136467Ssimokawa		return(drv_init);
500136467Ssimokawa
501136467Ssimokawa	drv_init = -1;
502136467Ssimokawa
503136467Ssimokawa	bzero(&dg, sizeof(dg));
504136467Ssimokawa	dcons_conf = &dg;
505136467Ssimokawa	dg.cdev = &dcons_consdev;
506136467Ssimokawa	dg.buf = NULL;
507136467Ssimokawa	dg.size = DCONS_BUF_SIZE;
508136467Ssimokawa
509136467Ssimokawa#ifdef __i386__
510136467Ssimokawa	if (getenv_int("dcons.addr", &addr) > 0 &&
511136467Ssimokawa	    getenv_int("dcons.size", &size) > 0) {
512136467Ssimokawa		/* XXX P to V */
513136467Ssimokawa		dg.buf = (struct dcons_buf *)(KERNBASE + addr);
514136467Ssimokawa		dg.size = size;
515136467Ssimokawa		if (dcons_load_buffer(dg.buf, dg.size, sc) < 0)
516136467Ssimokawa			dg.buf = NULL;
517136467Ssimokawa	}
518136467Ssimokawa#endif
519136467Ssimokawa	if (dg.buf != NULL)
520136467Ssimokawa		goto ok;
521136467Ssimokawa
522136467Ssimokawa#ifndef KLD_MODULE
523136467Ssimokawa	if (stage == 0) { /* XXX or cold */
524136467Ssimokawa		/*
525136467Ssimokawa		 * DCONS_FORCE_CONSOLE == 1 and statically linked.
526136467Ssimokawa		 * called from cninit(). can't use contigmalloc yet .
527136467Ssimokawa		 */
528136467Ssimokawa		dg.buf = (struct dcons_buf *) bssbuf;
529136467Ssimokawa		dcons_init(dg.buf, dg.size, sc);
530136467Ssimokawa	} else
531136467Ssimokawa#endif
532136467Ssimokawa	{
533136467Ssimokawa		/*
534136467Ssimokawa		 * DCONS_FORCE_CONSOLE == 0 or kernel module case.
535136467Ssimokawa		 * if the module is loaded after boot,
536136467Ssimokawa		 * bssbuf could be non-continuous.
537136467Ssimokawa		 */
538136467Ssimokawa		dg.buf = (struct dcons_buf *) contigmalloc(dg.size,
539136467Ssimokawa			M_DEVBUF, 0, 0x10000, 0xffffffff, PAGE_SIZE, 0ul);
540136467Ssimokawa		dcons_init(dg.buf, dg.size, sc);
541136467Ssimokawa	}
542136467Ssimokawa
543136467Ssimokawaok:
544136467Ssimokawa	dcons_buf = dg.buf;
545136467Ssimokawa
546136467Ssimokawa#if __FreeBSD_version < 502122
547136467Ssimokawa#if DDB && DCONS_FORCE_GDB
548136467Ssimokawa#if CONS_NODEV
549136467Ssimokawa	gdbconsdev.cn_arg = (void *)&sc[DCONS_GDB];
550136467Ssimokawa#if __FreeBSD_version >= 501109
551136467Ssimokawa	sprintf(gdbconsdev.cn_name, "dgdb");
552136467Ssimokawa#endif
553136467Ssimokawa	gdb_arg = &gdbconsdev;
554136467Ssimokawa#elif defined(__DragonFly__)
555136467Ssimokawa	gdbdev = make_dev(&dcons_cdevsw, DCONS_GDB,
556136467Ssimokawa	    UID_ROOT, GID_WHEEL, 0600, "dgdb");
557136467Ssimokawa#else
558136467Ssimokawa	gdbdev = makedev(CDEV_MAJOR, DCONS_GDB);
559136467Ssimokawa#endif
560136467Ssimokawa	gdb_getc = dcons_cngetc;
561136467Ssimokawa	gdb_putc = dcons_cnputc;
562136467Ssimokawa#endif
563136467Ssimokawa#endif
564136467Ssimokawa	drv_init = 1;
565136467Ssimokawa
566136467Ssimokawa	return 0;
567136467Ssimokawa}
568136467Ssimokawa
569136467Ssimokawa
570136467Ssimokawastatic int
571136467Ssimokawadcons_attach_port(int port, char *name, int flags)
572136467Ssimokawa{
573136467Ssimokawa	struct dcons_softc *dc;
574136467Ssimokawa	struct tty *tp;
575136467Ssimokawa	DEV dev;
576136467Ssimokawa
577136467Ssimokawa	dc = &sc[port];
578136467Ssimokawa	dc->flags = flags;
579136467Ssimokawa	dev = make_dev(&dcons_cdevsw, port,
580136467Ssimokawa			UID_ROOT, GID_WHEEL, 0600, name);
581136467Ssimokawa	dc->dev = (void *)dev;
582136467Ssimokawa	tp = ttymalloc(NULL);
583136467Ssimokawa
584136467Ssimokawa	dev->si_drv1 = (void *)dc;
585136467Ssimokawa	dev->si_tty = tp;
586136467Ssimokawa
587136467Ssimokawa	tp->t_oproc = dcons_tty_start;
588136467Ssimokawa	tp->t_param = dcons_tty_param;
589136467Ssimokawa	tp->t_stop = nottystop;
590136467Ssimokawa	tp->t_dev = dc->dev;
591136467Ssimokawa
592136467Ssimokawa	return(0);
593136467Ssimokawa}
594136467Ssimokawa
595136467Ssimokawastatic int
596136467Ssimokawadcons_attach(void)
597136467Ssimokawa{
598136467Ssimokawa	int polltime;
599136467Ssimokawa
600136467Ssimokawa#ifdef __DragonFly__
601136467Ssimokawa	cdevsw_add(&dcons_cdevsw, -1, 0);
602136467Ssimokawa#endif
603136467Ssimokawa	dcons_attach_port(DCONS_CON, "dcons", 0);
604136467Ssimokawa	dcons_attach_port(DCONS_GDB, "dgdb", DC_GDB);
605136467Ssimokawa#if __FreeBSD_version < 500000
606136467Ssimokawa	callout_init(&dcons_callout);
607136467Ssimokawa#else
608136467Ssimokawa	callout_init(&dcons_callout, 0);
609136467Ssimokawa#endif
610136467Ssimokawa	polltime = hz / poll_hz;
611136467Ssimokawa	if (polltime < 1)
612136467Ssimokawa		polltime = 1;
613136467Ssimokawa	callout_reset(&dcons_callout, polltime, dcons_timeout, NULL);
614136467Ssimokawa	return(0);
615136467Ssimokawa}
616136467Ssimokawa
617136467Ssimokawastatic int
618136467Ssimokawadcons_detach(int port)
619136467Ssimokawa{
620136467Ssimokawa	struct	tty *tp;
621136467Ssimokawa	struct dcons_softc *dc;
622136467Ssimokawa
623136467Ssimokawa	dc = &sc[port];
624136467Ssimokawa
625136467Ssimokawa	tp = ((DEV)dc->dev)->si_tty;
626136467Ssimokawa
627136467Ssimokawa	if (tp->t_state & TS_ISOPEN) {
628136467Ssimokawa		printf("dcons: still opened\n");
629136467Ssimokawa#if __FreeBSD_version < 502113
630136467Ssimokawa		(*linesw[tp->t_line].l_close)(tp, 0);
631136467Ssimokawa		tp->t_gen++;
632136467Ssimokawa		ttyclose(tp);
633136467Ssimokawa		ttwakeup(tp);
634136467Ssimokawa		ttwwakeup(tp);
635136467Ssimokawa#else
636136467Ssimokawa		ttyld_close(tp, 0);
637136467Ssimokawa		tty_close(tp);
638136467Ssimokawa#endif
639136467Ssimokawa	}
640136467Ssimokawa	/* XXX
641136467Ssimokawa	 * must wait until all device are closed.
642136467Ssimokawa	 */
643136467Ssimokawa#ifdef __DragonFly__
644136467Ssimokawa	tsleep((void *)dc, 0, "dcodtc", hz/4);
645136467Ssimokawa#else
646136467Ssimokawa	tsleep((void *)dc, PWAIT, "dcodtc", hz/4);
647136467Ssimokawa#endif
648136467Ssimokawa	destroy_dev(dc->dev);
649136467Ssimokawa
650136467Ssimokawa	return(0);
651136467Ssimokawa}
652136467Ssimokawa
653136467Ssimokawa
654136467Ssimokawa/* cnXXX works only for FreeBSD-5 */
655136467Ssimokawastatic int
656136467Ssimokawadcons_modevent(module_t mode, int type, void *data)
657136467Ssimokawa{
658136467Ssimokawa	int err = 0, ret;
659136467Ssimokawa
660136467Ssimokawa	switch (type) {
661136467Ssimokawa	case MOD_LOAD:
662136467Ssimokawa		ret = dcons_drv_init(1);
663136467Ssimokawa		dcons_attach();
664136467Ssimokawa#if __FreeBSD_version >= 500000
665136467Ssimokawa		if (ret == 0) {
666136467Ssimokawa			dcons_cnprobe(&dcons_consdev);
667136467Ssimokawa			dcons_cninit(&dcons_consdev);
668136467Ssimokawa			cnadd(&dcons_consdev);
669136467Ssimokawa		}
670136467Ssimokawa#endif
671136467Ssimokawa		break;
672136467Ssimokawa	case MOD_UNLOAD:
673136467Ssimokawa		printf("dcons: unload\n");
674136467Ssimokawa		callout_stop(&dcons_callout);
675136467Ssimokawa#if __FreeBSD_version < 502122
676136467Ssimokawa#if DDB && DCONS_FORCE_GDB
677136467Ssimokawa#if CONS_NODEV
678136467Ssimokawa		gdb_arg = NULL;
679136467Ssimokawa#else
680136467Ssimokawa		gdbdev = NULL;
681136467Ssimokawa#endif
682136467Ssimokawa#endif
683136467Ssimokawa#endif
684136467Ssimokawa#if __FreeBSD_version >= 500000
685136467Ssimokawa		cnremove(&dcons_consdev);
686136467Ssimokawa#endif
687136467Ssimokawa		dcons_detach(DCONS_CON);
688136467Ssimokawa		dcons_detach(DCONS_GDB);
689136467Ssimokawa		dg.buf->magic = 0;
690136467Ssimokawa
691136467Ssimokawa		contigfree(dg.buf, DCONS_BUF_SIZE, M_DEVBUF);
692136467Ssimokawa
693136467Ssimokawa		break;
694136467Ssimokawa	case MOD_SHUTDOWN:
695136467Ssimokawa		dg.buf->magic = 0;
696136467Ssimokawa		break;
697136467Ssimokawa	default:
698136467Ssimokawa		err = EOPNOTSUPP;
699136467Ssimokawa		break;
700136467Ssimokawa	}
701136467Ssimokawa	return(err);
702136467Ssimokawa}
703136467Ssimokawa
704136467Ssimokawa#if __FreeBSD_version >= 502122
705136467Ssimokawa/* Debugger interface */
706136467Ssimokawa
707136467Ssimokawastatic int
708136467Ssimokawadcons_dbg_probe(void)
709136467Ssimokawa{
710136467Ssimokawa	return(DCONS_FORCE_GDB);
711136467Ssimokawa}
712136467Ssimokawa
713136467Ssimokawastatic void
714136467Ssimokawadcons_dbg_init(void)
715136467Ssimokawa{
716136467Ssimokawa}
717136467Ssimokawa
718136467Ssimokawastatic void
719136467Ssimokawadcons_dbg_term(void)
720136467Ssimokawa{
721136467Ssimokawa}
722136467Ssimokawa
723136467Ssimokawastatic void
724136467Ssimokawadcons_dbg_putc(int c)
725136467Ssimokawa{
726136467Ssimokawa	struct dcons_softc *dc = &sc[DCONS_GDB];
727136467Ssimokawa	dcons_os_putc(dc, c);
728136467Ssimokawa}
729136467Ssimokawa
730136467Ssimokawastatic int
731136467Ssimokawadcons_dbg_checkc(void)
732136467Ssimokawa{
733136467Ssimokawa	struct dcons_softc *dc = &sc[DCONS_GDB];
734136467Ssimokawa	return (dcons_os_checkc(dc));
735136467Ssimokawa}
736136467Ssimokawa
737136467Ssimokawastatic int
738136467Ssimokawadcons_dbg_getc(void)
739136467Ssimokawa{
740136467Ssimokawa	struct dcons_softc *dc = &sc[DCONS_GDB];
741136467Ssimokawa	return (dcons_os_getc(dc));
742136467Ssimokawa}
743136467Ssimokawa#endif
744136467Ssimokawa
745136467SsimokawaDEV_MODULE(dcons, dcons_modevent, NULL);
746136467SsimokawaMODULE_VERSION(dcons, DCONS_VERSION);
747