mirror of
				https://gitlab.freedesktop.org/wayland/wayland.git
				synced 2025-11-03 09:01:42 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			544 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			544 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright © 2008 Kristian Høgsberg
 | 
						|
 *
 | 
						|
 * Permission to use, copy, modify, distribute, and sell this software and its
 | 
						|
 * documentation for any purpose is hereby granted without fee, provided that
 | 
						|
 * the above copyright notice appear in all copies and that both that copyright
 | 
						|
 * notice and this permission notice appear in supporting documentation, and
 | 
						|
 * that the name of the copyright holders not be used in advertising or
 | 
						|
 * publicity pertaining to distribution of the software without specific,
 | 
						|
 * written prior permission.  The copyright holders make no representations
 | 
						|
 * about the suitability of this software for any purpose.  It is provided "as
 | 
						|
 * is" without express or implied warranty.
 | 
						|
 *
 | 
						|
 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 | 
						|
 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
 | 
						|
 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
 | 
						|
 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
 | 
						|
 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
 | 
						|
 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
 | 
						|
 * OF THIS SOFTWARE.
 | 
						|
 */
 | 
						|
 | 
						|
#include <stdint.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <math.h>
 | 
						|
#include <time.h>
 | 
						|
#include <pty.h>
 | 
						|
#include <ctype.h>
 | 
						|
#include <cairo.h>
 | 
						|
#include <glib.h>
 | 
						|
#include <linux/input.h>
 | 
						|
 | 
						|
#include <GL/gl.h>
 | 
						|
#include <eagle.h>
 | 
						|
 | 
						|
#include "wayland-client.h"
 | 
						|
#include "wayland-glib.h"
 | 
						|
 | 
						|
#include "cairo-util.h"
 | 
						|
#include "window.h"
 | 
						|
 | 
						|
static const char gem_device[] = "/dev/dri/card0";
 | 
						|
static const char socket_name[] = "\0wayland";
 | 
						|
 | 
						|
#define MOD_SHIFT	0x01
 | 
						|
#define MOD_ALT		0x02
 | 
						|
#define MOD_CTRL	0x04
 | 
						|
 | 
						|
struct terminal {
 | 
						|
	struct window *window;
 | 
						|
	struct wl_display *display;
 | 
						|
	int redraw_scheduled, redraw_pending;
 | 
						|
	char *data;
 | 
						|
	int width, height, tail, row, column, total_rows;
 | 
						|
	int fd, master;
 | 
						|
	struct buffer *buffer;
 | 
						|
	GIOChannel *channel;
 | 
						|
	uint32_t modifiers;
 | 
						|
	char escape[64];
 | 
						|
	int escape_length;
 | 
						|
	int state;
 | 
						|
	int margin;
 | 
						|
};
 | 
						|
 | 
						|
static void
 | 
						|
terminal_resize(struct terminal *terminal, int width, int height)
 | 
						|
{
 | 
						|
	size_t size;
 | 
						|
	char *data;
 | 
						|
	int i, l, total_rows, row, tail;
 | 
						|
 | 
						|
	if (terminal->width == width && terminal->height == height)
 | 
						|
		return;
 | 
						|
 | 
						|
	size = (width + 1) * height;
 | 
						|
	data = malloc(size);
 | 
						|
	memset(data, 0, size);
 | 
						|
	if (terminal->data) {
 | 
						|
		if (width > terminal->width)
 | 
						|
			l = terminal->width;
 | 
						|
		else
 | 
						|
			l = width;
 | 
						|
 | 
						|
		if (terminal->total_rows > height) {
 | 
						|
			total_rows = height;
 | 
						|
			tail = terminal->tail + terminal->total_rows - height;
 | 
						|
		} else {
 | 
						|
			total_rows = terminal->total_rows;
 | 
						|
			tail = terminal->tail;
 | 
						|
		}
 | 
						|
 | 
						|
		for (i = 0; i < total_rows; i++) {
 | 
						|
			row = (tail + i) % terminal->height;
 | 
						|
			memcpy(data + (width + 1) * i,
 | 
						|
			       &terminal->data[row * (terminal->width + 1)], l);
 | 
						|
		}
 | 
						|
 | 
						|
		free(terminal->data);
 | 
						|
	} else {
 | 
						|
		total_rows = 1;
 | 
						|
	}
 | 
						|
 | 
						|
	terminal->width = width;
 | 
						|
	terminal->height = height;
 | 
						|
	terminal->data = data;
 | 
						|
 | 
						|
	terminal->total_rows = total_rows;
 | 
						|
	terminal->row = total_rows - 1;
 | 
						|
	if (terminal->column >= terminal->width)
 | 
						|
		terminal->column = terminal->width - 1;
 | 
						|
	terminal->tail = 0;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
terminal_draw_contents(struct terminal *terminal)
 | 
						|
{
 | 
						|
	struct rectangle rectangle;
 | 
						|
	cairo_surface_t *surface;
 | 
						|
	cairo_t *cr;
 | 
						|
	cairo_font_extents_t extents;
 | 
						|
	int i, row;
 | 
						|
 | 
						|
	window_get_child_rectangle(terminal->window, &rectangle);
 | 
						|
 | 
						|
	surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
 | 
						|
					     rectangle.width, rectangle.height);
 | 
						|
	cr = cairo_create(surface);
 | 
						|
	cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
 | 
						|
	cairo_set_source_rgba(cr, 0, 0, 0, 0.9);
 | 
						|
	cairo_paint(cr);
 | 
						|
	cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | 
						|
	cairo_set_source_rgba(cr, 0, 0.7, 0, 1);
 | 
						|
 | 
						|
	cairo_select_font_face (cr, "mono",
 | 
						|
				CAIRO_FONT_SLANT_NORMAL,
 | 
						|
				CAIRO_FONT_WEIGHT_NORMAL);
 | 
						|
	cairo_set_font_size(cr, 14);
 | 
						|
 | 
						|
	cairo_font_extents(cr, &extents);
 | 
						|
	for (i = 0; i < terminal->total_rows; i++) {
 | 
						|
		row = (terminal->tail + i) % terminal->height;
 | 
						|
		cairo_move_to(cr, terminal->margin,
 | 
						|
			      terminal->margin + extents.ascent + extents.height * i);
 | 
						|
		cairo_show_text(cr, &terminal->data[row * (terminal->width + 1)]);
 | 
						|
	}
 | 
						|
	cairo_destroy(cr);
 | 
						|
 | 
						|
	terminal->buffer = buffer_create_from_cairo_surface(terminal->fd, surface);
 | 
						|
	cairo_surface_destroy(surface);
 | 
						|
 | 
						|
	window_copy(terminal->window,
 | 
						|
		    &rectangle,
 | 
						|
		    terminal->buffer->name, terminal->buffer->stride);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
terminal_draw(struct terminal *terminal)
 | 
						|
{
 | 
						|
	struct rectangle rectangle;
 | 
						|
	cairo_surface_t *surface;
 | 
						|
	cairo_font_extents_t extents;
 | 
						|
	cairo_t *cr;
 | 
						|
	int width, height;
 | 
						|
 | 
						|
	window_get_child_rectangle(terminal->window, &rectangle);
 | 
						|
 | 
						|
	surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 0, 0);
 | 
						|
	cr = cairo_create(surface);
 | 
						|
	cairo_select_font_face (cr, "mono",
 | 
						|
				CAIRO_FONT_SLANT_NORMAL,
 | 
						|
				CAIRO_FONT_WEIGHT_NORMAL);
 | 
						|
	cairo_set_font_size(cr, 14);
 | 
						|
	cairo_font_extents(cr, &extents);
 | 
						|
	cairo_destroy(cr);
 | 
						|
	cairo_surface_destroy(surface);
 | 
						|
 | 
						|
	width = (rectangle.width - 2 * terminal->margin) / (int32_t) extents.max_x_advance;
 | 
						|
	height = (rectangle.height - 2 * terminal->margin) / (int32_t) extents.height;
 | 
						|
	terminal_resize(terminal, width, height);
 | 
						|
 | 
						|
	rectangle.width = terminal->width * extents.max_x_advance + 2 * terminal->margin;
 | 
						|
	rectangle.height = terminal->height * extents.height + 2 * terminal->margin;
 | 
						|
 | 
						|
	window_set_child_size(terminal->window, &rectangle);
 | 
						|
 | 
						|
	window_draw(terminal->window);
 | 
						|
	terminal_draw_contents(terminal);
 | 
						|
	wl_display_commit(terminal->display, 0);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
idle_redraw(void *data)
 | 
						|
{
 | 
						|
	struct terminal *terminal = data;
 | 
						|
 | 
						|
	terminal_draw(terminal);
 | 
						|
 | 
						|
	return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
#define STATE_NORMAL 0
 | 
						|
#define STATE_ESCAPE 1
 | 
						|
 | 
						|
static void
 | 
						|
terminal_data(struct terminal *terminal, const char *data, size_t length);
 | 
						|
 | 
						|
static void
 | 
						|
terminal_schedule_redraw(struct terminal *terminal)
 | 
						|
{
 | 
						|
	if (!terminal->redraw_scheduled) {
 | 
						|
		g_idle_add(idle_redraw, terminal);
 | 
						|
		terminal->redraw_scheduled = 1;
 | 
						|
	} else {
 | 
						|
		terminal->redraw_pending = 1;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
handle_escape(struct terminal *terminal)
 | 
						|
{
 | 
						|
	char *row;
 | 
						|
	int i, j;
 | 
						|
 | 
						|
	terminal->escape[terminal->escape_length++] = '\0';
 | 
						|
	if (strcmp(terminal->escape, "\e[J") == 0) {
 | 
						|
		row = &terminal->data[terminal->row * (terminal->width + 1)];
 | 
						|
		memset(&row[terminal->column], 0, terminal->width - terminal->column);
 | 
						|
		for (i = terminal->total_rows; i < terminal->height; i++) {
 | 
						|
 | 
						|
			j = terminal->row + i;
 | 
						|
			if (j >= terminal->height)
 | 
						|
				j -= terminal->height;
 | 
						|
			
 | 
						|
			row = &terminal->data[j * (terminal->width + 1)];
 | 
						|
			memset(row, 0, terminal->width);
 | 
						|
		}
 | 
						|
	} else if (strcmp(terminal->escape, "\e[H") == 0) {
 | 
						|
		terminal->row = terminal->tail;
 | 
						|
		terminal->total_rows = 1;
 | 
						|
		terminal->column = 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
terminal_data(struct terminal *terminal, const char *data, size_t length)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
	char *row;
 | 
						|
 | 
						|
	for (i = 0; i < length; i++) {
 | 
						|
		row = &terminal->data[terminal->row * (terminal->width + 1)];
 | 
						|
 | 
						|
		if (terminal->state == STATE_ESCAPE) {
 | 
						|
			terminal->escape[terminal->escape_length++] = data[i];
 | 
						|
			if (terminal->escape_length == 2 && data[i] != '[') {
 | 
						|
				/* Bad escape sequence. */
 | 
						|
				terminal->state = STATE_NORMAL;
 | 
						|
				goto cancel_escape;
 | 
						|
			}
 | 
						|
 | 
						|
			if (isalpha(data[i])) {
 | 
						|
				terminal->state = STATE_NORMAL;
 | 
						|
				handle_escape(terminal);
 | 
						|
			} 
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
	cancel_escape:
 | 
						|
		switch (data[i]) {
 | 
						|
		case '\r':
 | 
						|
			terminal->column = 0;
 | 
						|
			break;
 | 
						|
		case '\n':
 | 
						|
			terminal->column = 0;
 | 
						|
			terminal->row++;
 | 
						|
			if (terminal->row == terminal->height)
 | 
						|
				terminal->row = 0;
 | 
						|
			if (terminal->total_rows == terminal->height) {
 | 
						|
				memset(&terminal->data[terminal->row * (terminal->width + 1)],
 | 
						|
				       0, terminal->width);
 | 
						|
				terminal->tail++;
 | 
						|
			} else {
 | 
						|
				terminal->total_rows++;
 | 
						|
			}
 | 
						|
 | 
						|
			if (terminal->tail == terminal->height)
 | 
						|
				terminal->tail = 0;
 | 
						|
			break;
 | 
						|
		case '\t':
 | 
						|
			memset(&row[terminal->column], ' ', -terminal->column & 7);
 | 
						|
			terminal->column = (terminal->column + 7) & ~7;
 | 
						|
			break;
 | 
						|
		case '\e':
 | 
						|
			terminal->state = STATE_ESCAPE;
 | 
						|
			terminal->escape[0] = '\e';
 | 
						|
			terminal->escape_length = 1;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			if (terminal->column < terminal->width)
 | 
						|
				row[terminal->column++] = data[i];
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	terminal_schedule_redraw(terminal);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
resize_handler(struct window *window, void *data)
 | 
						|
{
 | 
						|
	struct terminal *terminal = data;
 | 
						|
 | 
						|
	terminal_schedule_redraw(terminal);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
acknowledge_handler(struct window *window, uint32_t key, void *data)
 | 
						|
{
 | 
						|
	struct terminal *terminal = data;
 | 
						|
 | 
						|
	terminal->redraw_scheduled = 0;
 | 
						|
	buffer_destroy(terminal->buffer, terminal->fd);
 | 
						|
 | 
						|
	if (terminal->redraw_pending) {
 | 
						|
		terminal->redraw_pending = 0;
 | 
						|
		terminal_schedule_redraw(terminal);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
struct key {
 | 
						|
	int code[4];
 | 
						|
} evdev_keymap[] = {
 | 
						|
	{ { 0, 0 } },		/* 0 */
 | 
						|
	{ { 0x1b, 0x1b } },
 | 
						|
	{ { '1', '!' } },
 | 
						|
	{ { '2', '@' } },
 | 
						|
	{ { '3', '#' } },
 | 
						|
	{ { '4', '$' } },
 | 
						|
	{ { '5', '%' } },
 | 
						|
	{ { '6', '^' } },
 | 
						|
	{ { '7', '&' } },
 | 
						|
	{ { '8', '*' } },
 | 
						|
	{ { '9', '(' } },
 | 
						|
	{ { '0', ')' } },
 | 
						|
	{ { '-', '_' } },
 | 
						|
	{ { '=', '+' } },
 | 
						|
	{ { '\b', '\b' } },
 | 
						|
	{ { '\t', '\t' } },
 | 
						|
 | 
						|
	{ { 'q', 'Q' } },		/* 16 */
 | 
						|
	{ { 'w', 'W', 0x17 } },
 | 
						|
	{ { 'e', 'E', 0x05 } },
 | 
						|
	{ { 'r', 'R', 0x12 } },
 | 
						|
	{ { 't', 'T' } },
 | 
						|
	{ { 'y', 'Y' } },
 | 
						|
	{ { 'u', 'U', 0x15 } },
 | 
						|
	{ { 'i', 'I' } },
 | 
						|
	{ { 'o', 'O' } },
 | 
						|
	{ { 'p', 'P', 0x10 } },
 | 
						|
	{ { '[', '{' } },
 | 
						|
	{ { ']', '}' } },
 | 
						|
	{ { '\n', '\n' } },
 | 
						|
	{ { 0, 0 } },
 | 
						|
	{ { 'a', 'A', 0x01} },
 | 
						|
	{ { 's', 'S', 0x13 } },
 | 
						|
 | 
						|
	{ { 'd', 'D', 0x04 } },		/* 32 */
 | 
						|
	{ { 'f', 'F', 0x06 } },
 | 
						|
	{ { 'g', 'G' } },
 | 
						|
	{ { 'h', 'H' } },
 | 
						|
	{ { 'j', 'J', 0x0a } },
 | 
						|
	{ { 'k', 'K', 0x0b } },
 | 
						|
	{ { 'l', 'L', 0x0c } },
 | 
						|
	{ { ';', ':' } },
 | 
						|
	{ { '\'', '"' } },
 | 
						|
	{ { '`', '~' } },
 | 
						|
	{ { 0, 0 } },
 | 
						|
	{ { '\\', '|' } },
 | 
						|
	{ { 'z', 'Z' } },
 | 
						|
	{ { 'x', 'X' } },
 | 
						|
	{ { 'c', 'C' } },
 | 
						|
	{ { 'v', 'V' } },
 | 
						|
 | 
						|
	{ { 'b', 'B', 0x02 } },		/* 48 */
 | 
						|
	{ { 'n', 'N', 0x0e } },
 | 
						|
	{ { 'm', 'M', 0x0d } },
 | 
						|
	{ { ',', '<' } },
 | 
						|
	{ { '.', '>' } },
 | 
						|
	{ { '/', '?' } },
 | 
						|
	{ { 0, 0 } },
 | 
						|
	{ { '*', '*' } },
 | 
						|
	{ { 0, 0 } },
 | 
						|
	{ { ' ', ' ' } },
 | 
						|
	{ { 0, 0 } }
 | 
						|
 | 
						|
	/* 59 */
 | 
						|
};
 | 
						|
 | 
						|
#define ARRAY_LENGTH(a) (sizeof (a) / sizeof (a)[0])
 | 
						|
 | 
						|
static void
 | 
						|
key_handler(struct window *window, uint32_t key, uint32_t state, void *data)
 | 
						|
{
 | 
						|
	struct terminal *terminal = data;
 | 
						|
	uint32_t mod = 0;
 | 
						|
	char c;
 | 
						|
 | 
						|
	switch (key) {
 | 
						|
	case KEY_LEFTSHIFT:
 | 
						|
	case KEY_RIGHTSHIFT:
 | 
						|
		mod = MOD_SHIFT;
 | 
						|
		break;
 | 
						|
	case KEY_LEFTCTRL:
 | 
						|
	case KEY_RIGHTCTRL:
 | 
						|
		mod = MOD_CTRL;
 | 
						|
		break;
 | 
						|
	case KEY_LEFTALT:
 | 
						|
	case KEY_RIGHTALT:
 | 
						|
		mod = MOD_ALT;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		if (key < ARRAY_LENGTH(evdev_keymap)) {
 | 
						|
			if (terminal->modifiers & MOD_CTRL)
 | 
						|
				c = evdev_keymap[key].code[2];
 | 
						|
			else if (terminal->modifiers & MOD_SHIFT)
 | 
						|
				c = evdev_keymap[key].code[1];
 | 
						|
			else
 | 
						|
				c = evdev_keymap[key].code[0];
 | 
						|
			if (state && c)
 | 
						|
				write(terminal->master, &c, 1);
 | 
						|
		}
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	if (state)
 | 
						|
		terminal->modifiers |= mod;
 | 
						|
	else
 | 
						|
		terminal->modifiers &= ~mod;
 | 
						|
}
 | 
						|
 | 
						|
static struct terminal *
 | 
						|
terminal_create(struct wl_display *display, int fd)
 | 
						|
{
 | 
						|
	struct terminal *terminal;
 | 
						|
 | 
						|
	terminal = malloc(sizeof *terminal);
 | 
						|
	if (terminal == NULL)
 | 
						|
		return terminal;
 | 
						|
 | 
						|
	memset(terminal, 0, sizeof *terminal);
 | 
						|
	terminal->fd = fd;
 | 
						|
	terminal->window = window_create(display, fd, "Wayland Terminal",
 | 
						|
					 500, 100, 500, 400);
 | 
						|
	terminal->display = display;
 | 
						|
	terminal->redraw_scheduled = 1;
 | 
						|
	terminal->margin = 5;
 | 
						|
 | 
						|
	window_set_resize_handler(terminal->window, resize_handler, terminal);
 | 
						|
	window_set_acknowledge_handler(terminal->window, acknowledge_handler, terminal);
 | 
						|
	window_set_key_handler(terminal->window, key_handler, terminal);
 | 
						|
 | 
						|
	terminal_draw(terminal);
 | 
						|
 | 
						|
	return terminal;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
io_handler(GIOChannel   *source,
 | 
						|
	   GIOCondition  condition,
 | 
						|
	   gpointer      data)
 | 
						|
{
 | 
						|
	struct terminal *terminal = data;
 | 
						|
	gchar buffer[256];
 | 
						|
	gsize bytes_read;
 | 
						|
	GError *error = NULL;
 | 
						|
 | 
						|
	g_io_channel_read_chars(source, buffer, sizeof buffer,
 | 
						|
				&bytes_read, &error);
 | 
						|
 | 
						|
	terminal_data(terminal, buffer, bytes_read);
 | 
						|
 | 
						|
	return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
terminal_run(struct terminal *terminal, const char *path)
 | 
						|
{
 | 
						|
	int master, slave;
 | 
						|
	pid_t pid;
 | 
						|
 | 
						|
	pid = forkpty(&master, NULL, NULL, NULL);
 | 
						|
	if (pid == 0) {
 | 
						|
		close(master);
 | 
						|
		if (execl(path, path, NULL)) {
 | 
						|
			printf("exec failed: %m\n");
 | 
						|
			exit(EXIT_FAILURE);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	close(slave);
 | 
						|
	terminal->master = master;
 | 
						|
	terminal->channel = g_io_channel_unix_new(master);
 | 
						|
	fcntl(master, F_SETFL, O_NONBLOCK);
 | 
						|
	g_io_add_watch(terminal->channel, G_IO_IN,
 | 
						|
		       io_handler, terminal);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int main(int argc, char *argv[])
 | 
						|
{
 | 
						|
	struct wl_display *display;
 | 
						|
	int fd;
 | 
						|
	GMainLoop *loop;
 | 
						|
	GSource *source;
 | 
						|
	struct terminal *terminal;
 | 
						|
 | 
						|
	fd = open(gem_device, O_RDWR);
 | 
						|
	if (fd < 0) {
 | 
						|
		fprintf(stderr, "drm open failed: %m\n");
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	display = wl_display_create(socket_name, sizeof socket_name);
 | 
						|
	if (display == NULL) {
 | 
						|
		fprintf(stderr, "failed to create display: %m\n");
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	loop = g_main_loop_new(NULL, FALSE);
 | 
						|
	source = wl_glib_source_new(display);
 | 
						|
	g_source_attach(source, NULL);
 | 
						|
 | 
						|
	terminal = terminal_create(display, fd);
 | 
						|
	terminal_run(terminal, "/bin/bash");
 | 
						|
 | 
						|
	g_main_loop_run(loop);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 |