NetBSD/sys/arch/usermode/dev/vncfb.c
2012-01-04 14:53:57 +00:00

717 lines
18 KiB
C

/* $NetBSD: vncfb.c,v 1.12 2012/01/04 14:53:57 jmcneill Exp $ */
/*-
* Copyright (c) 2011 Jared D. McNeill <jmcneill@invisible.ca>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by Jared D. McNeill.
* 4. Neither the name of The NetBSD Foundation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include "opt_wsemul.h"
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: vncfb.c,v 1.12 2012/01/04 14:53:57 jmcneill Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/device.h>
#include <sys/kmem.h>
#include <uvm/uvm_extern.h>
#include <dev/wscons/wsconsio.h>
#include <dev/wscons/wsdisplayvar.h>
#include <dev/wsfont/wsfont.h>
#include <dev/rasops/rasops.h>
#include <dev/wscons/wsdisplay_vconsvar.h>
#include <dev/wscons/wskbdvar.h>
#include <dev/wscons/wsksymdef.h>
#include <dev/wscons/wsksymvar.h>
#include <dev/wscons/wsmousevar.h>
#include <machine/mainbus.h>
#include <machine/thunk.h>
#define VNCFB_REFRESH_INTERVAL 33 /* fb refresh interval when mapped */
struct vncfb_fbops {
void (*copycols)(void *, int, int, int, int);
void (*erasecols)(void *, int, int, int, long);
void (*copyrows)(void *, int, int, int);
void (*eraserows)(void *, int, int, long);
void (*putchar)(void *, int, int, u_int, long);
void (*cursor)(void *, int, int, int);
};
struct vncfb_softc {
device_t sc_dev;
device_t sc_wskbddev;
device_t sc_wsmousedev;
thunk_rfb_t sc_rfb;
unsigned int sc_width;
unsigned int sc_height;
unsigned int sc_depth;
int sc_mode;
uint8_t * sc_mem;
size_t sc_memsize;
uint8_t * sc_framebuf;
size_t sc_framebufsize;
struct vcons_data sc_vd;
struct vncfb_fbops sc_ops;
int sc_kbd_enable;
int sc_mouse_enable;
void *sc_ih;
void *sc_sih;
callout_t sc_callout;
void *sc_refresh_sih;
};
static int vncfb_match(device_t, cfdata_t, void *);
static void vncfb_attach(device_t, device_t, void *);
CFATTACH_DECL_NEW(vncfb, sizeof(struct vncfb_softc),
vncfb_match, vncfb_attach, NULL, NULL);
static void vncfb_putchar(void *, int, int, u_int, long);
static void vncfb_copycols(void *, int, int, int, int);
static void vncfb_erasecols(void *, int, int, int, long);
static void vncfb_copyrows(void *, int, int, int);
static void vncfb_eraserows(void *, int, int, long);
static void vncfb_cursor(void *, int, int, int);
static int vncfb_ioctl(void *, void *, u_long, void *, int, lwp_t *);
static paddr_t vncfb_mmap(void *, void *, off_t, int);
static void vncfb_init_screen(void *, struct vcons_screen *, int, long *);
static void vncfb_update(struct vncfb_softc *, int, int, int, int);
static void vncfb_copyrect(struct vncfb_softc *, int, int, int, int, int, int);
static void vncfb_fillrect(struct vncfb_softc *, int, int, int, int, uint32_t);
static int vncfb_intr(void *);
static void vncfb_softintr(void *);
static void vncfb_refresh(void *);
static void vncfb_softrefresh(void *);
static int vncfb_kbd_enable(void *, int);
static void vncfb_kbd_set_leds(void *, int);
static int vncfb_kbd_ioctl(void *, u_long, void *, int, lwp_t *);
static void vncfb_kbd_cngetc(void *, u_int *, int *);
static void vncfb_kbd_cnpollc(void *, int);
static void vncfb_kbd_bell(void *, u_int, u_int, u_int);
static int vncfb_mouse_enable(void *);
static int vncfb_mouse_ioctl(void *, u_long, void *, int, lwp_t *);
static void vncfb_mouse_disable(void *);
static struct vcons_screen vncfb_console_screen;
static struct wsscreen_descr vncfb_defaultscreen = {
.name = "default",
.fontwidth = 8,
.fontheight = 16,
.capabilities = WSSCREEN_WSCOLORS | WSSCREEN_HILIT,
};
static const struct wsscreen_descr *vncfb_screens[] = {
&vncfb_defaultscreen,
};
static struct wsscreen_list vncfb_screenlist = {
.screens = vncfb_screens,
.nscreens = __arraycount(vncfb_screens),
};
static struct wsdisplay_accessops vncfb_accessops = {
.ioctl = vncfb_ioctl,
.mmap = vncfb_mmap,
};
extern const struct wscons_keydesc vnckbd_keydesctab[];
static const struct wskbd_mapdata vncfb_keymapdata = {
vnckbd_keydesctab,
KB_US,
};
static struct wskbd_accessops vncfb_kbd_accessops = {
vncfb_kbd_enable,
vncfb_kbd_set_leds,
vncfb_kbd_ioctl,
};
static const struct wskbd_consops vncfb_kbd_consops = {
vncfb_kbd_cngetc,
vncfb_kbd_cnpollc,
vncfb_kbd_bell,
};
static const struct wsmouse_accessops vncfb_mouse_accessops = {
vncfb_mouse_enable,
vncfb_mouse_ioctl,
vncfb_mouse_disable,
};
static int
vncfb_match(device_t parent, cfdata_t match, void *priv)
{
struct thunkbus_attach_args *taa = priv;
return taa->taa_type == THUNKBUS_TYPE_VNCFB;
}
static void
vncfb_attach(device_t parent, device_t self, void *priv)
{
struct vncfb_softc *sc = device_private(self);
struct thunkbus_attach_args *taa = priv;
struct wsemuldisplaydev_attach_args waa;
struct wskbddev_attach_args kaa;
struct wsmousedev_attach_args maa;
struct rasops_info *ri;
unsigned long defattr;
sc->sc_dev = self;
sc->sc_width = taa->u.vnc.width;
sc->sc_height = taa->u.vnc.height;
sc->sc_depth = 32;
sc->sc_mode = WSDISPLAYIO_MODE_EMUL;
#if notyet
sc->sc_sockfd = thunk_vnc_open_socket(taa->u.vnc.port);
if (sc->sc_sockfd == -1)
panic("couldn't open VNC socket");
#endif
sc->sc_framebufsize = sc->sc_width * sc->sc_height * (sc->sc_depth / 8);
sc->sc_memsize = sc->sc_framebufsize + PAGE_SIZE;
sc->sc_mem = kmem_zalloc(sc->sc_memsize, KM_SLEEP);
sc->sc_framebuf = (void *)round_page((vaddr_t)sc->sc_mem);
aprint_naive("\n");
aprint_normal(": %ux%u %ubpp (port %u)\n",
sc->sc_width, sc->sc_height, sc->sc_depth, taa->u.vnc.port);
sc->sc_rfb.width = sc->sc_width;
sc->sc_rfb.height = sc->sc_height;
sc->sc_rfb.depth = sc->sc_depth;
sc->sc_rfb.framebuf = sc->sc_framebuf;
snprintf(sc->sc_rfb.name, sizeof(sc->sc_rfb.name),
"NetBSD/usermode %d.%d.%d",
__NetBSD_Version__ / 100000000,
(__NetBSD_Version__ / 1000000) % 100,
(__NetBSD_Version__ / 100) % 100);
if (thunk_rfb_open(&sc->sc_rfb, taa->u.vnc.port) != 0)
panic("couldn't open rfb server");
sc->sc_sih = softint_establish(SOFTINT_SERIAL, vncfb_softintr, sc);
sc->sc_ih = sigio_intr_establish(vncfb_intr, sc);
sc->sc_refresh_sih = softint_establish(SOFTINT_SERIAL,
vncfb_softrefresh, sc);
callout_init(&sc->sc_callout, 0);
callout_setfunc(&sc->sc_callout, vncfb_refresh, sc);
vcons_init(&sc->sc_vd, sc, &vncfb_defaultscreen, &vncfb_accessops);
sc->sc_vd.init_screen = vncfb_init_screen;
ri = &vncfb_console_screen.scr_ri;
vcons_init_screen(&sc->sc_vd, &vncfb_console_screen, 1, &defattr);
vncfb_console_screen.scr_flags |= VCONS_SCREEN_IS_STATIC;
vncfb_defaultscreen.textops = &ri->ri_ops;
vncfb_defaultscreen.capabilities = ri->ri_caps;
vncfb_defaultscreen.nrows = ri->ri_rows;
vncfb_defaultscreen.ncols = ri->ri_cols;
wsdisplay_cnattach(&vncfb_defaultscreen, ri, 0, 0, defattr);
vcons_replay_msgbuf(&vncfb_console_screen);
waa.console = true;
waa.scrdata = &vncfb_screenlist;
waa.accessops = &vncfb_accessops;
waa.accesscookie = &sc->sc_vd;
config_found(self, &waa, wsemuldisplaydevprint);
wskbd_cnattach(&vncfb_kbd_consops, sc, &vncfb_keymapdata);
kaa.console = true;
kaa.keymap = &vncfb_keymapdata;
kaa.accessops = &vncfb_kbd_accessops;
kaa.accesscookie = sc;
sc->sc_wskbddev = config_found_ia(self, "wskbddev", &kaa,
wskbddevprint);
maa.accessops = &vncfb_mouse_accessops;
maa.accesscookie = sc;
sc->sc_wsmousedev = config_found_ia(self, "wsmousedev", &maa,
wsmousedevprint);
}
static void
vncfb_init_screen(void *priv, struct vcons_screen *scr, int existing,
long *defattr)
{
struct vncfb_softc *sc = priv;
struct vncfb_fbops *ops = &sc->sc_ops;
struct rasops_info *ri = &scr->scr_ri;
ri->ri_width = sc->sc_width;
ri->ri_height = sc->sc_height;
ri->ri_depth = sc->sc_depth;
ri->ri_stride = sc->sc_width * ri->ri_depth / 8;
ri->ri_bits = sc->sc_framebuf;
ri->ri_flg = RI_CENTER | RI_FULLCLEAR;
if (existing)
ri->ri_flg |= RI_CLEAR;
rasops_init(ri, sc->sc_height / 8, sc->sc_width / 8);
ri->ri_caps = WSSCREEN_WSCOLORS;
rasops_reconfig(ri, sc->sc_height / ri->ri_font->fontheight,
sc->sc_width / ri->ri_font->fontwidth);
ri->ri_hw = scr;
ops->putchar = ri->ri_ops.putchar;
ops->copyrows = ri->ri_ops.copyrows;
ops->eraserows = ri->ri_ops.eraserows;
ops->copycols = ri->ri_ops.copycols;
ops->erasecols = ri->ri_ops.erasecols;
ops->cursor = ri->ri_ops.cursor;
ri->ri_ops.copyrows = vncfb_copyrows;
ri->ri_ops.copycols = vncfb_copycols;
ri->ri_ops.eraserows = vncfb_eraserows;
ri->ri_ops.erasecols = vncfb_erasecols;
ri->ri_ops.putchar = vncfb_putchar;
ri->ri_ops.cursor = vncfb_cursor;
}
static void
vncfb_putchar(void *priv, int row, int col, u_int c, long attr)
{
struct rasops_info *ri = priv;
struct vcons_screen *scr = ri->ri_hw;
struct vncfb_softc *sc = scr->scr_cookie;
struct vncfb_fbops *ops = &sc->sc_ops;
int x, y, w, h;
ops->putchar(ri, row, col, c, attr);
x = ri->ri_xorigin + (col * ri->ri_font->fontwidth);
y = ri->ri_yorigin + (row * ri->ri_font->fontheight);
w = ri->ri_font->fontwidth;
h = ri->ri_font->fontheight;
vncfb_update(sc, x, y, w, h);
}
static void
vncfb_copycols(void *priv, int row, int srccol, int dstcol, int ncols)
{
struct rasops_info *ri = priv;
struct vcons_screen *scr = ri->ri_hw;
struct vncfb_softc *sc = scr->scr_cookie;
struct vncfb_fbops *ops = &sc->sc_ops;
int x, y, w, h;
ops->copycols(ri, row, srccol, dstcol, ncols);
y = ri->ri_yorigin + (row * ri->ri_font->fontheight);
h = ri->ri_font->fontheight;
if (srccol < dstcol) {
x = ri->ri_xorigin + (srccol * ri->ri_font->fontwidth);
w = (dstcol - srccol) * ri->ri_font->fontwidth;
} else {
x = ri->ri_xorigin + (dstcol * ri->ri_font->fontwidth);
w = (srccol - dstcol) * ri->ri_font->fontwidth;
}
vncfb_update(sc, x, y, w, h);
}
static void
vncfb_erasecols(void *priv, int row, int startcol, int ncols, long fillattr)
{
struct rasops_info *ri = priv;
struct vcons_screen *scr = ri->ri_hw;
struct vncfb_softc *sc = scr->scr_cookie;
struct vncfb_fbops *ops = &sc->sc_ops;
int x, y, w, h, c;
ops->erasecols(ri, row, startcol, ncols, fillattr);
y = ri->ri_yorigin + (row * ri->ri_font->fontheight);
h = ri->ri_font->fontheight;
x = ri->ri_xorigin + (startcol * ri->ri_font->fontwidth);
w = ncols * ri->ri_font->fontwidth;
c = ri->ri_devcmap[(fillattr >> 16) & 0xf] & 0xffffff;
vncfb_fillrect(sc, x, y, w, h, c);
}
static void
vncfb_copyrows(void *priv, int srcrow, int dstrow, int nrows)
{
struct rasops_info *ri = priv;
struct vcons_screen *scr = ri->ri_hw;
struct vncfb_softc *sc = scr->scr_cookie;
struct vncfb_fbops *ops = &sc->sc_ops;
int x, y, w, h, srcx, srcy;
int fontheight;
/* barrier */
while (sc->sc_rfb.nupdates > 0)
if (thunk_rfb_poll(&sc->sc_rfb, NULL) == -1)
break;
ops->copyrows(ri, srcrow, dstrow, nrows);
fontheight = ri->ri_font->fontheight;
x = ri->ri_xorigin;
y = ri->ri_yorigin + dstrow * fontheight;
w = ri->ri_width;
h = nrows * fontheight;
srcx = ri->ri_xorigin;
srcy = ri->ri_yorigin + srcrow * fontheight;
vncfb_copyrect(sc, x, y, w, h, srcx, srcy);
}
static void
vncfb_eraserows(void *priv, int row, int nrows, long fillattr)
{
struct rasops_info *ri = priv;
struct vcons_screen *scr = ri->ri_hw;
struct vncfb_softc *sc = scr->scr_cookie;
struct vncfb_fbops *ops = &sc->sc_ops;
int x, y, w, h, c;
ops->eraserows(ri, row, nrows, fillattr);
y = ri->ri_yorigin + (row * ri->ri_font->fontheight);
h = nrows * ri->ri_font->fontheight;
x = ri->ri_xorigin;
w = ri->ri_width;
c = ri->ri_devcmap[(fillattr >> 16) & 0xf] & 0xffffff;
vncfb_fillrect(sc, x, y, w, h, c);
}
static void
vncfb_cursor(void *priv, int on, int row, int col)
{
struct rasops_info *ri = priv;
struct vcons_screen *scr = ri->ri_hw;
struct vncfb_softc *sc = scr->scr_cookie;
struct vncfb_fbops *ops = &sc->sc_ops;
int ox, oy, x, y, w, h;
w = ri->ri_font->fontwidth;
h = ri->ri_font->fontheight;
ox = ri->ri_ccol * w + ri->ri_xorigin;
oy = ri->ri_crow * h + ri->ri_yorigin;
ops->cursor(ri, on, row, col);
x = ri->ri_ccol * w + ri->ri_xorigin;
y = ri->ri_crow * h + ri->ri_yorigin;
vncfb_update(sc, ox, oy, w, h);
vncfb_update(sc, x, y, w, h);
}
static int
vncfb_ioctl(void *v, void *vs, u_long cmd, void *data, int flag, lwp_t *l)
{
struct vcons_data *vd = v;
struct vncfb_softc *sc = vd->cookie;
struct wsdisplay_fbinfo *wdf;
struct vcons_screen *ms = vd->active;
int new_mode;
switch (cmd) {
case WSDISPLAYIO_GTYPE:
*(u_int *)data = WSDISPLAY_TYPE_VNC;
return 0;
case WSDISPLAYIO_GINFO:
wdf = data;
wdf->height = ms->scr_ri.ri_height;
wdf->width = ms->scr_ri.ri_width;
wdf->depth = ms->scr_ri.ri_depth;
wdf->cmsize = 256;
return 0;
case WSDISPLAYIO_LINEBYTES:
*(u_int *)data = sc->sc_width * (sc->sc_depth / 8);
return 0;
case WSDISPLAYIO_SMODE:
new_mode = *(int *)data;
if (sc->sc_mode != new_mode) {
sc->sc_mode = new_mode;
if (new_mode == WSDISPLAYIO_MODE_EMUL) {
callout_halt(&sc->sc_callout, NULL);
vcons_redraw_screen(ms);
} else {
callout_schedule(&sc->sc_callout, 1);
}
}
return 0;
default:
return EPASSTHROUGH;
}
}
static paddr_t
vncfb_mmap(void *v, void *vs, off_t offset, int prot)
{
struct vcons_data *vd = v;
struct vncfb_softc *sc = vd->cookie;
paddr_t pa;
vaddr_t va;
if (offset < 0 || offset >= sc->sc_framebufsize) {
device_printf(sc->sc_dev, "mmap: offset 0x%x, fbsize 0x%x"
" out of range!\n",
(unsigned int)offset, (unsigned int)sc->sc_framebufsize);
return -1;
}
va = trunc_page((vaddr_t)sc->sc_framebuf + offset);
if (pmap_extract(pmap_kernel(), va, &pa) == false) {
device_printf(sc->sc_dev, "mmap: pmap_extract failed!\n");
return -1;
}
return atop(pa);
}
static void
vncfb_update(struct vncfb_softc *sc, int x, int y, int w, int h)
{
thunk_rfb_update(&sc->sc_rfb, x, y, w, h);
softint_schedule(sc->sc_sih);
}
static void
vncfb_copyrect(struct vncfb_softc *sc, int x, int y, int w, int h,
int srcx, int srcy)
{
thunk_rfb_copyrect(&sc->sc_rfb, x, y, w, h, srcx, srcy);
softint_schedule(sc->sc_sih);
}
static void
vncfb_fillrect(struct vncfb_softc *sc, int x, int y, int w, int h, uint32_t c)
{
thunk_rfb_fillrect(&sc->sc_rfb, x, y, w, h, (uint8_t *)&c);
softint_schedule(sc->sc_sih);
}
static int
vncfb_intr(void *priv)
{
struct vncfb_softc *sc = priv;
softint_schedule(sc->sc_sih);
return 0;
}
static void
vncfb_softintr(void *priv)
{
struct vncfb_softc *sc = priv;
thunk_rfb_event_t event;
int s;
while (thunk_rfb_poll(&sc->sc_rfb, &event) > 0) {
switch (event.message_type) {
case THUNK_RFB_KEY_EVENT:
s = spltty();
wskbd_input(sc->sc_wskbddev,
event.data.key_event.down_flag ?
WSCONS_EVENT_KEY_DOWN : WSCONS_EVENT_KEY_UP,
event.data.key_event.keysym & 0xfff);
splx(s);
break;
case THUNK_RFB_POINTER_EVENT:
s = spltty();
wsmouse_input(sc->sc_wsmousedev,
event.data.pointer_event.button_mask,
event.data.pointer_event.absx,
event.data.pointer_event.absy,
0, 0,
WSMOUSE_INPUT_ABSOLUTE_X|WSMOUSE_INPUT_ABSOLUTE_Y);
splx(s);
default:
break;
}
}
}
static void
vncfb_refresh(void *priv)
{
struct vncfb_softc *sc = priv;
softint_schedule(sc->sc_refresh_sih);
}
static void
vncfb_softrefresh(void *priv)
{
struct vncfb_softc *sc = priv;
if (sc->sc_mode == WSDISPLAYIO_MODE_EMUL)
return;
/* update the screen */
vncfb_update(sc, 0, 0, sc->sc_width, sc->sc_height);
/* flush the pending drawing op */
while (thunk_rfb_poll(&sc->sc_rfb, NULL) > 0)
;
callout_schedule(&sc->sc_callout, mstohz(VNCFB_REFRESH_INTERVAL));
}
static int
vncfb_kbd_enable(void *priv, int on)
{
struct vncfb_softc *sc = priv;
sc->sc_kbd_enable = on;
return 0;
}
static void
vncfb_kbd_set_leds(void *priv, int leds)
{
}
static int
vncfb_kbd_ioctl(void *priv, u_long cmd, void *data, int flag, lwp_t *l)
{
struct wskbd_bell_data *bd;
switch (cmd) {
case WSKBDIO_GTYPE:
*(int *)data = WSKBD_TYPE_RFB;
return 0;
case WSKBDIO_COMPLEXBELL:
bd = data;
vncfb_kbd_bell(priv, bd->pitch, bd->period, bd->volume);
return 0;
default:
return EPASSTHROUGH;
}
}
static void
vncfb_kbd_cngetc(void *priv, u_int *type, int *data)
{
struct vncfb_softc *sc = priv;
thunk_rfb_event_t event;
for (;;) {
if (thunk_rfb_poll(&sc->sc_rfb, &event) > 0) {
if (event.message_type == THUNK_RFB_KEY_EVENT) {
*type = event.data.key_event.down_flag ?
WSCONS_EVENT_KEY_DOWN : WSCONS_EVENT_KEY_UP;
*data = event.data.key_event.keysym & 0xfff;
return;
}
}
}
}
static void
vncfb_kbd_cnpollc(void *priv, int on)
{
struct vncfb_softc *sc = priv;
if (!on) {
vncfb_intr(sc);
}
}
static void
vncfb_kbd_bell(void *priv, u_int pitch, u_int period, u_int volume)
{
struct vncfb_softc *sc = priv;
thunk_rfb_bell(&sc->sc_rfb);
softint_schedule(sc->sc_sih);
}
static int
vncfb_mouse_enable(void *priv)
{
struct vncfb_softc *sc = priv;
sc->sc_mouse_enable = 1;
return 0;
}
static int
vncfb_mouse_ioctl(void *priv, u_long cmd, void *data, int flag, lwp_t *l)
{
switch (cmd) {
case WSMOUSEIO_GTYPE:
*(u_int *)data = WSMOUSE_TYPE_PSEUDO;
return 0;
default:
return EPASSTHROUGH;
}
}
static void
vncfb_mouse_disable(void *priv)
{
struct vncfb_softc *sc = priv;
sc->sc_mouse_enable = 0;
}