line.c 19.0 KB
Newer Older
1
/*
J
Jeff Dike 已提交
2
 * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
L
Linus Torvalds 已提交
3 4 5
 * Licensed under the GPL
 */

J
Jeff Dike 已提交
6
#include "linux/irqreturn.h"
L
Linus Torvalds 已提交
7 8
#include "linux/kd.h"
#include "chan_kern.h"
J
Jeff Dike 已提交
9
#include "irq_kern.h"
L
Linus Torvalds 已提交
10 11 12 13 14
#include "irq_user.h"
#include "os.h"

#define LINE_BUFSIZE 4096

A
Al Viro 已提交
15
static irqreturn_t line_interrupt(int irq, void *data)
L
Linus Torvalds 已提交
16
{
17 18 19
	struct chan *chan = data;
	struct line *line = chan->line;
	struct tty_struct *tty = line->tty;
L
Linus Torvalds 已提交
20 21 22 23 24 25

	if (line)
		chan_interrupt(&line->chan_list, &line->task, tty, irq);
	return IRQ_HANDLED;
}

A
Andrew Morton 已提交
26
static void line_timer_cb(struct work_struct *work)
L
Linus Torvalds 已提交
27
{
A
Andrew Morton 已提交
28
	struct line *line = container_of(work, struct line, task.work);
L
Linus Torvalds 已提交
29

J
Jeff Dike 已提交
30
	if (!line->throttled)
31 32
		chan_interrupt(&line->chan_list, &line->task, line->tty,
			       line->driver->read_irq);
L
Linus Torvalds 已提交
33 34
}

J
Jeff Dike 已提交
35 36
/*
 * Returns the free space inside the ring buffer of this line.
37
 *
S
Simon Arlott 已提交
38
 * Should be called while holding line->lock (this does not modify data).
39 40
 */
static int write_room(struct line *line)
L
Linus Torvalds 已提交
41 42 43
{
	int n;

44 45 46 47 48
	if (line->buffer == NULL)
		return LINE_BUFSIZE - 1;

	/* This is for the case where the buffer is wrapped! */
	n = line->head - line->tail;
L
Linus Torvalds 已提交
49 50

	if (n <= 0)
J
Jeff Dike 已提交
51
		n += LINE_BUFSIZE; /* The other case */
52 53 54 55 56 57 58 59 60 61 62 63 64 65
	return n - 1;
}

int line_write_room(struct tty_struct *tty)
{
	struct line *line = tty->driver_data;
	unsigned long flags;
	int room;

	spin_lock_irqsave(&line->lock, flags);
	room = write_room(line);
	spin_unlock_irqrestore(&line->lock, flags);

	return room;
L
Linus Torvalds 已提交
66 67
}

68 69 70 71 72 73 74
int line_chars_in_buffer(struct tty_struct *tty)
{
	struct line *line = tty->driver_data;
	unsigned long flags;
	int ret;

	spin_lock_irqsave(&line->lock, flags);
J
Jeff Dike 已提交
75
	/* write_room subtracts 1 for the needed NULL, so we readd it.*/
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
	ret = LINE_BUFSIZE - (write_room(line) + 1);
	spin_unlock_irqrestore(&line->lock, flags);

	return ret;
}

/*
 * This copies the content of buf into the circular buffer associated with
 * this line.
 * The return value is the number of characters actually copied, i.e. the ones
 * for which there was space: this function is not supposed to ever flush out
 * the circular buffer.
 *
 * Must be called while holding line->lock!
 */
L
Linus Torvalds 已提交
91 92 93 94
static int buffer_data(struct line *line, const char *buf, int len)
{
	int end, room;

J
Jeff Dike 已提交
95
	if (line->buffer == NULL) {
L
Linus Torvalds 已提交
96 97
		line->buffer = kmalloc(LINE_BUFSIZE, GFP_ATOMIC);
		if (line->buffer == NULL) {
J
Jeff Dike 已提交
98 99 100
			printk(KERN_ERR "buffer_data - atomic allocation "
			       "failed\n");
			return 0;
L
Linus Torvalds 已提交
101 102 103 104 105 106 107 108 109
		}
		line->head = line->buffer;
		line->tail = line->buffer;
	}

	room = write_room(line);
	len = (len > room) ? room : len;

	end = line->buffer + LINE_BUFSIZE - line->tail;
110

J
Jeff Dike 已提交
111
	if (len < end) {
L
Linus Torvalds 已提交
112 113
		memcpy(line->tail, buf, len);
		line->tail += len;
J
Jeff Dike 已提交
114 115
	}
	else {
116
		/* The circular buffer is wrapping */
L
Linus Torvalds 已提交
117 118 119 120 121 122
		memcpy(line->tail, buf, end);
		buf += end;
		memcpy(line->buffer, buf, len - end);
		line->tail = line->buffer + len - end;
	}

123
	return len;
L
Linus Torvalds 已提交
124 125
}

126 127 128 129 130 131 132 133 134
/*
 * Flushes the ring buffer to the output channels. That is, write_chan is
 * called, passing it line->head as buffer, and an appropriate count.
 *
 * On exit, returns 1 when the buffer is empty,
 * 0 when the buffer is not empty on exit,
 * and -errno when an error occurred.
 *
 * Must be called while holding line->lock!*/
L
Linus Torvalds 已提交
135 136 137 138 139
static int flush_buffer(struct line *line)
{
	int n, count;

	if ((line->buffer == NULL) || (line->head == line->tail))
140
		return 1;
L
Linus Torvalds 已提交
141 142

	if (line->tail < line->head) {
143
		/* line->buffer + LINE_BUFSIZE is the end of the buffer! */
L
Linus Torvalds 已提交
144
		count = line->buffer + LINE_BUFSIZE - line->head;
145

L
Linus Torvalds 已提交
146 147 148
		n = write_chan(&line->chan_list, line->head, count,
			       line->driver->write_irq);
		if (n < 0)
149 150
			return n;
		if (n == count) {
J
Jeff Dike 已提交
151 152 153 154
			/*
			 * We have flushed from ->head to buffer end, now we
			 * must flush only from the beginning to ->tail.
			 */
L
Linus Torvalds 已提交
155
			line->head = line->buffer;
156
		} else {
L
Linus Torvalds 已提交
157
			line->head += n;
158
			return 0;
L
Linus Torvalds 已提交
159 160 161 162
		}
	}

	count = line->tail - line->head;
J
Jeff Dike 已提交
163
	n = write_chan(&line->chan_list, line->head, count,
L
Linus Torvalds 已提交
164
		       line->driver->write_irq);
165

J
Jeff Dike 已提交
166
	if (n < 0)
167
		return n;
L
Linus Torvalds 已提交
168 169

	line->head += n;
170 171 172 173 174 175 176 177 178 179 180 181 182 183
	return line->head == line->tail;
}

void line_flush_buffer(struct tty_struct *tty)
{
	struct line *line = tty->driver_data;
	unsigned long flags;
	int err;

	spin_lock_irqsave(&line->lock, flags);
	err = flush_buffer(line);
	spin_unlock_irqrestore(&line->lock, flags);
}

J
Jeff Dike 已提交
184 185 186 187
/*
 * We map both ->flush_chars and ->put_char (which go in pair) onto
 * ->flush_buffer and ->write. Hope it's not that bad.
 */
188 189 190 191 192 193 194 195
void line_flush_chars(struct tty_struct *tty)
{
	line_flush_buffer(tty);
}

void line_put_char(struct tty_struct *tty, unsigned char ch)
{
	line_write(tty, &ch, sizeof(ch));
L
Linus Torvalds 已提交
196 197 198 199 200 201
}

int line_write(struct tty_struct *tty, const unsigned char *buf, int len)
{
	struct line *line = tty->driver_data;
	unsigned long flags;
J
Jeff Dike 已提交
202
	int n, ret = 0;
L
Linus Torvalds 已提交
203

204
	spin_lock_irqsave(&line->lock, flags);
J
Jeff Dike 已提交
205
	if (line->head != line->tail)
L
Linus Torvalds 已提交
206
		ret = buffer_data(line, buf, len);
J
Jeff Dike 已提交
207
	else {
J
Jeff Dike 已提交
208
		n = write_chan(&line->chan_list, buf, len,
L
Linus Torvalds 已提交
209
			       line->driver->write_irq);
210
		if (n < 0) {
L
Linus Torvalds 已提交
211 212 213 214 215 216
			ret = n;
			goto out_up;
		}

		len -= n;
		ret += n;
217
		if (len > 0)
L
Linus Torvalds 已提交
218 219
			ret += buffer_data(line, buf + n, len);
	}
220 221 222
out_up:
	spin_unlock_irqrestore(&line->lock, flags);
	return ret;
L
Linus Torvalds 已提交
223 224
}

A
Alan Cox 已提交
225
void line_set_termios(struct tty_struct *tty, struct ktermios * old)
L
Linus Torvalds 已提交
226 227 228 229
{
	/* nothing */
}

J
Jeff Dike 已提交
230
static const struct {
L
Linus Torvalds 已提交
231 232 233 234 235 236
	int  cmd;
	char *level;
	char *name;
} tty_ioctls[] = {
	/* don't print these, they flood the log ... */
	{ TCGETS,      NULL,       "TCGETS"      },
J
Jeff Dike 已提交
237 238 239 240
	{ TCSETS,      NULL,       "TCSETS"      },
	{ TCSETSW,     NULL,       "TCSETSW"     },
	{ TCFLSH,      NULL,       "TCFLSH"      },
	{ TCSBRK,      NULL,       "TCSBRK"      },
L
Linus Torvalds 已提交
241 242

	/* general tty stuff */
J
Jeff Dike 已提交
243 244 245 246 247
	{ TCSETSF,     KERN_DEBUG, "TCSETSF"     },
	{ TCGETA,      KERN_DEBUG, "TCGETA"      },
	{ TIOCMGET,    KERN_DEBUG, "TIOCMGET"    },
	{ TCSBRKP,     KERN_DEBUG, "TCSBRKP"     },
	{ TIOCMSET,    KERN_DEBUG, "TIOCMSET"    },
L
Linus Torvalds 已提交
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303

	/* linux-specific ones */
	{ TIOCLINUX,   KERN_INFO,  "TIOCLINUX"   },
	{ KDGKBMODE,   KERN_INFO,  "KDGKBMODE"   },
	{ KDGKBTYPE,   KERN_INFO,  "KDGKBTYPE"   },
	{ KDSIGACCEPT, KERN_INFO,  "KDSIGACCEPT" },
};

int line_ioctl(struct tty_struct *tty, struct file * file,
	       unsigned int cmd, unsigned long arg)
{
	int ret;
	int i;

	ret = 0;
	switch(cmd) {
#ifdef TIOCGETP
	case TIOCGETP:
	case TIOCSETP:
	case TIOCSETN:
#endif
#ifdef TIOCGETC
	case TIOCGETC:
	case TIOCSETC:
#endif
#ifdef TIOCGLTC
	case TIOCGLTC:
	case TIOCSLTC:
#endif
	case TCGETS:
	case TCSETSF:
	case TCSETSW:
	case TCSETS:
	case TCGETA:
	case TCSETAF:
	case TCSETAW:
	case TCSETA:
	case TCXONC:
	case TCFLSH:
	case TIOCOUTQ:
	case TIOCINQ:
	case TIOCGLCKTRMIOS:
	case TIOCSLCKTRMIOS:
	case TIOCPKT:
	case TIOCGSOFTCAR:
	case TIOCSSOFTCAR:
		return -ENOIOCTLCMD;
#if 0
	case TCwhatever:
		/* do something */
		break;
#endif
	default:
		for (i = 0; i < ARRAY_SIZE(tty_ioctls); i++)
			if (cmd == tty_ioctls[i].cmd)
				break;
J
Jeff Dike 已提交
304
		if (i == ARRAY_SIZE(tty_ioctls)) {
L
Linus Torvalds 已提交
305 306 307 308 309 310
			printk(KERN_ERR "%s: %s: unknown ioctl: 0x%x\n",
			       __FUNCTION__, tty->name, cmd);
		}
		ret = -ENOIOCTLCMD;
		break;
	}
311
	return ret;
L
Linus Torvalds 已提交
312 313
}

314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
void line_throttle(struct tty_struct *tty)
{
	struct line *line = tty->driver_data;

	deactivate_chan(&line->chan_list, line->driver->read_irq);
	line->throttled = 1;
}

void line_unthrottle(struct tty_struct *tty)
{
	struct line *line = tty->driver_data;

	line->throttled = 0;
	chan_interrupt(&line->chan_list, &line->task, tty,
		       line->driver->read_irq);

J
Jeff Dike 已提交
330 331
	/*
	 * Maybe there is enough stuff pending that calling the interrupt
332 333 334
	 * throttles us again.  In this case, line->throttled will be 1
	 * again and we shouldn't turn the interrupt back on.
	 */
J
Jeff Dike 已提交
335
	if (!line->throttled)
336 337 338
		reactivate_chan(&line->chan_list, line->driver->read_irq);
}

A
Al Viro 已提交
339
static irqreturn_t line_write_interrupt(int irq, void *data)
L
Linus Torvalds 已提交
340
{
341 342 343
	struct chan *chan = data;
	struct line *line = chan->line;
	struct tty_struct *tty = line->tty;
L
Linus Torvalds 已提交
344 345
	int err;

J
Jeff Dike 已提交
346 347 348 349
	/*
	 * Interrupts are disabled here because we registered the interrupt with
	 * IRQF_DISABLED (see line_setup_irq).
	 */
350

351
	spin_lock(&line->lock);
L
Linus Torvalds 已提交
352
	err = flush_buffer(line);
353 354
	if (err == 0) {
		return IRQ_NONE;
J
Jeff Dike 已提交
355
	} else if (err < 0) {
L
Linus Torvalds 已提交
356 357 358
		line->head = line->buffer;
		line->tail = line->buffer;
	}
359
	spin_unlock(&line->lock);
L
Linus Torvalds 已提交
360

J
Jeff Dike 已提交
361
	if (tty == NULL)
362
		return IRQ_NONE;
L
Linus Torvalds 已提交
363

364
	if (test_bit(TTY_DO_WRITE_WAKEUP, &tty->flags) &&
L
Linus Torvalds 已提交
365 366
	   (tty->ldisc.write_wakeup != NULL))
		(tty->ldisc.write_wakeup)(tty);
367

J
Jeff Dike 已提交
368 369
	/*
	 * BLOCKING mode
L
Linus Torvalds 已提交
370 371 372 373 374
	 * In blocking mode, everything sleeps on tty->write_wait.
	 * Sleeping in the console driver would break non-blocking
	 * writes.
	 */

375
	if (waitqueue_active(&tty->write_wait))
L
Linus Torvalds 已提交
376
		wake_up_interruptible(&tty->write_wait);
377
	return IRQ_HANDLED;
L
Linus Torvalds 已提交
378 379
}

380
int line_setup_irq(int fd, int input, int output, struct line *line, void *data)
L
Linus Torvalds 已提交
381
{
J
Jeff Dike 已提交
382
	const struct line_driver *driver = line->driver;
383
	int err = 0, flags = IRQF_DISABLED | IRQF_SHARED | IRQF_SAMPLE_RANDOM;
L
Linus Torvalds 已提交
384

385 386
	if (input)
		err = um_request_irq(driver->read_irq, fd, IRQ_READ,
J
Jeff Dike 已提交
387
				       line_interrupt, flags,
388
				       driver->read_irq_name, data);
389 390 391 392
	if (err)
		return err;
	if (output)
		err = um_request_irq(driver->write_irq, fd, IRQ_WRITE,
J
Jeff Dike 已提交
393
					line_write_interrupt, flags,
394
					driver->write_irq_name, data);
L
Linus Torvalds 已提交
395
	line->have_irq = 1;
396
	return err;
L
Linus Torvalds 已提交
397 398
}

J
Jeff Dike 已提交
399 400
/*
 * Normally, a driver like this can rely mostly on the tty layer
J
Jeff Dike 已提交
401 402 403 404
 * locking, particularly when it comes to the driver structure.
 * However, in this case, mconsole requests can come in "from the
 * side", and race with opens and closes.
 *
405 406 407 408 409
 * mconsole config requests will want to be sure the device isn't in
 * use, and get_config, open, and close will want a stable
 * configuration.  The checking and modification of the configuration
 * is done under a spinlock.  Checking whether the device is in use is
 * line->tty->count > 1, also under the spinlock.
J
Jeff Dike 已提交
410
 *
411 412 413
 * tty->count serves to decide whether the device should be enabled or
 * disabled on the host.  If it's equal to 1, then we are doing the
 * first open or last close.  Otherwise, open and close just return.
J
Jeff Dike 已提交
414 415
 */

416
int line_open(struct line *lines, struct tty_struct *tty)
L
Linus Torvalds 已提交
417
{
J
Jeff Dike 已提交
418
	struct line *line = &lines[tty->index];
419
	int err = -ENODEV;
L
Linus Torvalds 已提交
420

J
Jeff Dike 已提交
421
	spin_lock(&line->count_lock);
J
Jeff Dike 已提交
422
	if (!line->valid)
J
Jeff Dike 已提交
423
		goto out_unlock;
L
Linus Torvalds 已提交
424

J
Jeff Dike 已提交
425
	err = 0;
J
Jeff Dike 已提交
426
	if (tty->count > 1)
J
Jeff Dike 已提交
427 428 429
		goto out_unlock;

	spin_unlock(&line->count_lock);
430

431 432 433
	tty->driver_data = line;
	line->tty = tty;

434 435 436 437
	err = enable_chan(line);
	if (err)
		return err;

J
Jeff Dike 已提交
438
	INIT_DELAYED_WORK(&line->task, line_timer_cb);
439

J
Jeff Dike 已提交
440
	if (!line->sigio) {
J
Jeff Dike 已提交
441 442
		chan_enable_winch(&line->chan_list, tty);
		line->sigio = 1;
L
Linus Torvalds 已提交
443 444
	}

J
Jeff Dike 已提交
445 446 447 448 449 450 451
	chan_window_size(&line->chan_list, &tty->winsize.ws_row,
			 &tty->winsize.ws_col);

	return err;

out_unlock:
	spin_unlock(&line->count_lock);
452
	return err;
L
Linus Torvalds 已提交
453 454
}

J
Jeff Dike 已提交
455 456
static void unregister_winch(struct tty_struct *tty);

L
Linus Torvalds 已提交
457 458 459 460
void line_close(struct tty_struct *tty, struct file * filp)
{
	struct line *line = tty->driver_data;

J
Jeff Dike 已提交
461 462
	/*
	 * If line_open fails (and tty->driver_data is never set),
J
Jeff Dike 已提交
463 464
	 * tty_open will call line_close.  So just return in this case.
	 */
J
Jeff Dike 已提交
465
	if (line == NULL)
J
Jeff Dike 已提交
466
		return;
467 468 469 470

	/* We ignore the error anyway! */
	flush_buffer(line);

J
Jeff Dike 已提交
471
	spin_lock(&line->count_lock);
J
Jeff Dike 已提交
472
	if (!line->valid)
J
Jeff Dike 已提交
473
		goto out_unlock;
J
Jeff Dike 已提交
474

J
Jeff Dike 已提交
475
	if (tty->count > 1)
J
Jeff Dike 已提交
476 477 478 479 480 481 482
		goto out_unlock;

	spin_unlock(&line->count_lock);

	line->tty = NULL;
	tty->driver_data = NULL;

J
Jeff Dike 已提交
483
	if (line->sigio) {
J
Jeff Dike 已提交
484 485
		unregister_winch(tty);
		line->sigio = 0;
J
Jeff Dike 已提交
486
	}
J
Jeff Dike 已提交
487

J
Jeff Dike 已提交
488 489 490 491
	return;

out_unlock:
	spin_unlock(&line->count_lock);
L
Linus Torvalds 已提交
492 493 494 495 496 497 498
}

void close_lines(struct line *lines, int nlines)
{
	int i;

	for(i = 0; i < nlines; i++)
499
		close_chan(&lines[i].chan_list, 0);
L
Linus Torvalds 已提交
500 501
}

502 503
static int setup_one_line(struct line *lines, int n, char *init, int init_prio,
			  char **error_out)
J
Jeff Dike 已提交
504 505
{
	struct line *line = &lines[n];
506
	int err = -EINVAL;
J
Jeff Dike 已提交
507 508 509

	spin_lock(&line->count_lock);

J
Jeff Dike 已提交
510
	if (line->tty != NULL) {
511
		*error_out = "Device is already open";
J
Jeff Dike 已提交
512 513 514
		goto out;
	}

J
Jeff Dike 已提交
515
	if (line->init_pri <= init_prio) {
J
Jeff Dike 已提交
516 517 518 519 520 521 522 523
		line->init_pri = init_prio;
		if (!strcmp(init, "none"))
			line->valid = 0;
		else {
			line->init_str = init;
			line->valid = 1;
		}
	}
524
	err = 0;
J
Jeff Dike 已提交
525 526
out:
	spin_unlock(&line->count_lock);
527
	return err;
J
Jeff Dike 已提交
528 529
}

J
Jeff Dike 已提交
530 531
/*
 * Common setup code for both startup command line and mconsole initialization.
M
Matt LaPlante 已提交
532
 * @lines contains the array (of size @num) to modify;
533
 * @init is the setup string;
534
 * @error_out is an error string in the case of failure;
535
 */
536

537 538
int line_setup(struct line *lines, unsigned int num, char *init,
	       char **error_out)
L
Linus Torvalds 已提交
539
{
540
	int i, n, err;
L
Linus Torvalds 已提交
541 542
	char *end;

J
Jeff Dike 已提交
543 544 545 546 547
	if (*init == '=') {
		/*
		 * We said con=/ssl= instead of con#=, so we are configuring all
		 * consoles at once.
		 */
548
		n = -1;
J
Jeff Dike 已提交
549 550
	}
	else {
L
Linus Torvalds 已提交
551
		n = simple_strtoul(init, &end, 0);
J
Jeff Dike 已提交
552
		if (*end != '=') {
553 554
			*error_out = "Couldn't parse device number";
			return -EINVAL;
L
Linus Torvalds 已提交
555 556 557 558
		}
		init = end;
	}
	init++;
559 560

	if (n >= (signed int) num) {
561 562 563
		*error_out = "Device number out of range";
		return -EINVAL;
	}
J
Jeff Dike 已提交
564
	else if (n >= 0) {
565
		err = setup_one_line(lines, n, init, INIT_ONE, error_out);
J
Jeff Dike 已提交
566
		if (err)
567
			return err;
J
Jeff Dike 已提交
568 569
	}
	else {
J
Jeff Dike 已提交
570
		for(i = 0; i < num; i++) {
571 572
			err = setup_one_line(lines, i, init, INIT_ALL,
					     error_out);
J
Jeff Dike 已提交
573
			if (err)
574 575
				return err;
		}
L
Linus Torvalds 已提交
576
	}
577
	return n == -1 ? num : n;
L
Linus Torvalds 已提交
578 579
}

580
int line_config(struct line *lines, unsigned int num, char *str,
581
		const struct chan_opts *opts, char **error_out)
L
Linus Torvalds 已提交
582
{
583
	struct line *line;
J
Jeff Dike 已提交
584
	char *new;
585
	int n;
L
Linus Torvalds 已提交
586

J
Jeff Dike 已提交
587
	if (*str == '=') {
588 589
		*error_out = "Can't configure all devices from mconsole";
		return -EINVAL;
590 591
	}

J
Jeff Dike 已提交
592
	new = kstrdup(str, GFP_KERNEL);
J
Jeff Dike 已提交
593
	if (new == NULL) {
594 595
		*error_out = "Failed to allocate memory";
		return -ENOMEM;
L
Linus Torvalds 已提交
596
	}
597
	n = line_setup(lines, num, new, error_out);
J
Jeff Dike 已提交
598
	if (n < 0)
599
		return n;
600 601

	line = &lines[n];
602
	return parse_chan_pair(line->init_str, line, n, opts, error_out);
L
Linus Torvalds 已提交
603 604
}

605
int line_get_config(char *name, struct line *lines, unsigned int num, char *str,
L
Linus Torvalds 已提交
606 607 608 609 610 611 612
		    int size, char **error_out)
{
	struct line *line;
	char *end;
	int dev, n = 0;

	dev = simple_strtoul(name, &end, 0);
J
Jeff Dike 已提交
613
	if ((*end != '\0') || (end == name)) {
L
Linus Torvalds 已提交
614
		*error_out = "line_get_config failed to parse device number";
615
		return 0;
L
Linus Torvalds 已提交
616 617
	}

J
Jeff Dike 已提交
618
	if ((dev < 0) || (dev >= num)) {
619 620
		*error_out = "device number out of range";
		return 0;
L
Linus Torvalds 已提交
621 622 623 624
	}

	line = &lines[dev];

J
Jeff Dike 已提交
625
	spin_lock(&line->count_lock);
J
Jeff Dike 已提交
626
	if (!line->valid)
L
Linus Torvalds 已提交
627
		CONFIG_CHUNK(str, size, n, "none", 1);
J
Jeff Dike 已提交
628
	else if (line->tty == NULL)
L
Linus Torvalds 已提交
629 630
		CONFIG_CHUNK(str, size, n, line->init_str, 1);
	else n = chan_config_string(&line->chan_list, str, size, error_out);
J
Jeff Dike 已提交
631
	spin_unlock(&line->count_lock);
L
Linus Torvalds 已提交
632

633
	return n;
L
Linus Torvalds 已提交
634 635
}

J
Jeff Dike 已提交
636 637 638
int line_id(char **str, int *start_out, int *end_out)
{
	char *end;
J
Jeff Dike 已提交
639
	int n;
J
Jeff Dike 已提交
640 641

	n = simple_strtoul(*str, &end, 0);
J
Jeff Dike 已提交
642 643
	if ((*end != '\0') || (end == *str))
		return -1;
J
Jeff Dike 已提交
644

J
Jeff Dike 已提交
645 646 647 648
	*str = end;
	*start_out = n;
	*end_out = n;
	return n;
J
Jeff Dike 已提交
649 650
}

651
int line_remove(struct line *lines, unsigned int num, int n, char **error_out)
L
Linus Torvalds 已提交
652
{
653
	int err;
L
Linus Torvalds 已提交
654 655
	char config[sizeof("conxxxx=none\0")];

J
Jeff Dike 已提交
656
	sprintf(config, "%d=none", n);
657
	err = line_setup(lines, num, config, error_out);
J
Jeff Dike 已提交
658
	if (err >= 0)
659 660
		err = 0;
	return err;
L
Linus Torvalds 已提交
661 662
}

663 664 665
struct tty_driver *register_lines(struct line_driver *line_driver,
				  const struct tty_operations *ops,
				  struct line *lines, int nlines)
L
Linus Torvalds 已提交
666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683
{
	int i;
	struct tty_driver *driver = alloc_tty_driver(nlines);

	if (!driver)
		return NULL;

	driver->driver_name = line_driver->name;
	driver->name = line_driver->device_name;
	driver->major = line_driver->major;
	driver->minor_start = line_driver->minor_start;
	driver->type = line_driver->type;
	driver->subtype = line_driver->subtype;
	driver->flags = TTY_DRIVER_REAL_RAW;
	driver->init_termios = tty_std_termios;
	tty_set_operations(driver, ops);

	if (tty_register_driver(driver)) {
J
Jeff Dike 已提交
684 685
		printk(KERN_ERR "register_lines : can't register %s driver\n",
		       line_driver->name);
L
Linus Torvalds 已提交
686 687 688 689
		put_tty_driver(driver);
		return NULL;
	}

J
Jeff Dike 已提交
690 691
	for(i = 0; i < nlines; i++) {
		if (!lines[i].valid)
L
Linus Torvalds 已提交
692 693 694 695 696 697 698
			tty_unregister_device(driver, i);
	}

	mconsole_register_dev(&line_driver->mc);
	return driver;
}

699 700
static DEFINE_SPINLOCK(winch_handler_lock);
static LIST_HEAD(winch_handlers);
701

702
void lines_init(struct line *lines, int nlines, struct chan_opts *opts)
L
Linus Torvalds 已提交
703 704
{
	struct line *line;
705
	char *error;
L
Linus Torvalds 已提交
706 707
	int i;

J
Jeff Dike 已提交
708
	for(i = 0; i < nlines; i++) {
L
Linus Torvalds 已提交
709 710
		line = &lines[i];
		INIT_LIST_HEAD(&line->chan_list);
711

J
Jeff Dike 已提交
712
		if (line->init_str == NULL)
J
Jeff Dike 已提交
713 714 715
			continue;

		line->init_str = kstrdup(line->init_str, GFP_KERNEL);
J
Jeff Dike 已提交
716 717
		if (line->init_str == NULL)
			printk(KERN_ERR "lines_init - kstrdup returned NULL\n");
718

J
Jeff Dike 已提交
719 720 721
		if (parse_chan_pair(line->init_str, line, i, opts, &error)) {
			printk(KERN_ERR "parse_chan_pair failed for "
			       "device %d : %s\n", i, error);
722 723
			line->valid = 0;
		}
L
Linus Torvalds 已提交
724 725 726 727 728 729 730 731 732
	}
}

struct winch {
	struct list_head list;
	int fd;
	int tty_fd;
	int pid;
	struct tty_struct *tty;
J
Jeff Dike 已提交
733
	unsigned long stack;
L
Linus Torvalds 已提交
734 735
};

J
Jeff Dike 已提交
736 737 738 739 740 741 742 743 744 745 746 747 748 749 750
static void free_winch(struct winch *winch, int free_irq_ok)
{
	list_del(&winch->list);

	if (winch->pid != -1)
		os_kill_process(winch->pid, 1);
	if (winch->fd != -1)
		os_close_file(winch->fd);
	if (winch->stack != 0)
		free_stack(winch->stack, 0);
	if (free_irq_ok)
		free_irq(WINCH_IRQ, winch);
	kfree(winch);
}

A
Al Viro 已提交
751
static irqreturn_t winch_interrupt(int irq, void *data)
L
Linus Torvalds 已提交
752 753 754 755 756 757 758
{
	struct winch *winch = data;
	struct tty_struct *tty;
	struct line *line;
	int err;
	char c;

J
Jeff Dike 已提交
759
	if (winch->fd != -1) {
L
Linus Torvalds 已提交
760
		err = generic_read(winch->fd, &c, NULL);
J
Jeff Dike 已提交
761 762 763 764 765 766
		if (err < 0) {
			if (err != -EAGAIN) {
				printk(KERN_ERR "winch_interrupt : "
				       "read failed, errno = %d\n", -err);
				printk(KERN_ERR "fd %d is losing SIGWINCH "
				       "support\n", winch->tty_fd);
J
Jeff Dike 已提交
767
				free_winch(winch, 0);
768
				return IRQ_HANDLED;
L
Linus Torvalds 已提交
769 770 771 772
			}
			goto out;
		}
	}
J
Jeff Dike 已提交
773
	tty = winch->tty;
L
Linus Torvalds 已提交
774 775
	if (tty != NULL) {
		line = tty->driver_data;
J
Jeff Dike 已提交
776
		chan_window_size(&line->chan_list, &tty->winsize.ws_row,
L
Linus Torvalds 已提交
777
				 &tty->winsize.ws_col);
778
		kill_pgrp(tty->pgrp, SIGWINCH, 1);
L
Linus Torvalds 已提交
779 780
	}
 out:
J
Jeff Dike 已提交
781
	if (winch->fd != -1)
L
Linus Torvalds 已提交
782
		reactivate_fd(winch->fd, WINCH_IRQ);
783
	return IRQ_HANDLED;
L
Linus Torvalds 已提交
784 785
}

J
Jeff Dike 已提交
786 787
void register_winch_irq(int fd, int tty_fd, int pid, struct tty_struct *tty,
			unsigned long stack)
L
Linus Torvalds 已提交
788 789 790 791 792
{
	struct winch *winch;

	winch = kmalloc(sizeof(*winch), GFP_KERNEL);
	if (winch == NULL) {
J
Jeff Dike 已提交
793
		printk(KERN_ERR "register_winch_irq - kmalloc failed\n");
J
Jeff Dike 已提交
794
		goto cleanup;
L
Linus Torvalds 已提交
795
	}
796

L
Linus Torvalds 已提交
797 798 799 800
	*winch = ((struct winch) { .list  	= LIST_HEAD_INIT(winch->list),
				   .fd  	= fd,
				   .tty_fd 	= tty_fd,
				   .pid  	= pid,
J
Jeff Dike 已提交
801 802 803 804 805 806
				   .tty 	= tty,
				   .stack	= stack });

	if (um_request_irq(WINCH_IRQ, fd, IRQ_READ, winch_interrupt,
			   IRQF_DISABLED | IRQF_SHARED | IRQF_SAMPLE_RANDOM,
			   "winch", winch) < 0) {
J
Jeff Dike 已提交
807 808
		printk(KERN_ERR "register_winch_irq - failed to register "
		       "IRQ\n");
J
Jeff Dike 已提交
809 810
		goto out_free;
	}
811 812

	spin_lock(&winch_handler_lock);
L
Linus Torvalds 已提交
813
	list_add(&winch->list, &winch_handlers);
814 815
	spin_unlock(&winch_handler_lock);

J
Jeff Dike 已提交
816
	return;
817

J
Jeff Dike 已提交
818
 out_free:
819
	kfree(winch);
J
Jeff Dike 已提交
820 821 822 823 824
 cleanup:
	os_kill_process(pid, 1);
	os_close_file(fd);
	if (stack != 0)
		free_stack(stack, 0);
825 826
}

J
Jeff Dike 已提交
827 828 829
static void unregister_winch(struct tty_struct *tty)
{
	struct list_head *ele;
830
	struct winch *winch;
J
Jeff Dike 已提交
831

832
	spin_lock(&winch_handler_lock);
833

J
Jeff Dike 已提交
834
	list_for_each(ele, &winch_handlers) {
J
Jeff Dike 已提交
835
		winch = list_entry(ele, struct winch, list);
J
Jeff Dike 已提交
836
		if (winch->tty == tty) {
J
Jeff Dike 已提交
837
			free_winch(winch, 1);
838
			break;
J
Jeff Dike 已提交
839 840
		}
	}
841
	spin_unlock(&winch_handler_lock);
J
Jeff Dike 已提交
842 843
}

L
Linus Torvalds 已提交
844 845
static void winch_cleanup(void)
{
846
	struct list_head *ele, *next;
L
Linus Torvalds 已提交
847 848
	struct winch *winch;

849 850
	spin_lock(&winch_handler_lock);

J
Jeff Dike 已提交
851
	list_for_each_safe(ele, next, &winch_handlers) {
L
Linus Torvalds 已提交
852
		winch = list_entry(ele, struct winch, list);
J
Jeff Dike 已提交
853
		free_winch(winch, 1);
L
Linus Torvalds 已提交
854
	}
855 856

	spin_unlock(&winch_handler_lock);
L
Linus Torvalds 已提交
857 858 859 860 861 862 863 864
}
__uml_exitcall(winch_cleanup);

char *add_xterm_umid(char *base)
{
	char *umid, *title;
	int len;

J
Jeff Dike 已提交
865
	umid = get_umid();
J
Jeff Dike 已提交
866
	if (*umid == '\0')
867
		return base;
868

L
Linus Torvalds 已提交
869 870
	len = strlen(base) + strlen(" ()") + strlen(umid) + 1;
	title = kmalloc(len, GFP_KERNEL);
J
Jeff Dike 已提交
871 872
	if (title == NULL) {
		printk(KERN_ERR "Failed to allocate buffer for xterm title\n");
873
		return base;
L
Linus Torvalds 已提交
874 875 876
	}

	snprintf(title, len, "%s (%s)", base, umid);
877
	return title;
L
Linus Torvalds 已提交
878
}