Home | History | Annotate | Line # | Download | only in common
ite_sti.c revision 1.2.6.1
      1  1.2.6.1  perseant /*	$NetBSD: ite_sti.c,v 1.2.6.1 2025/08/02 05:55:39 perseant Exp $	*/
      2      1.1   tsutsui /*	$OpenBSD: ite_sti.c,v 1.2 2011/08/18 20:02:58 miod Exp $	*/
      3      1.1   tsutsui /*
      4      1.1   tsutsui  * Copyright (c) 2006, 2011, Miodrag Vallat
      5      1.1   tsutsui  * Copyright (c) 2000-2003 Michael Shalayeff
      6      1.1   tsutsui  * All rights reserved.
      7      1.1   tsutsui  *
      8      1.1   tsutsui  * Redistribution and use in source and binary forms, with or without
      9      1.1   tsutsui  * modification, are permitted provided that the following conditions
     10      1.1   tsutsui  * are met:
     11      1.1   tsutsui  * 1. Redistributions of source code must retain the above copyright
     12      1.1   tsutsui  *    notice, this list of conditions and the following disclaimer.
     13      1.1   tsutsui  * 2. Redistributions in binary form must reproduce the above copyright
     14      1.1   tsutsui  *    notice, this list of conditions and the following disclaimer in the
     15      1.1   tsutsui  *    documentation and/or other materials provided with the distribution.
     16      1.1   tsutsui  *
     17      1.1   tsutsui  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     18      1.1   tsutsui  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     19      1.1   tsutsui  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     20      1.1   tsutsui  * IN NO EVENT SHALL THE AUTHOR OR HIS RELATIVES BE LIABLE FOR ANY DIRECT,
     21      1.1   tsutsui  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
     22      1.1   tsutsui  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
     23      1.1   tsutsui  * SERVICES; LOSS OF MIND, USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     24      1.1   tsutsui  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
     25      1.1   tsutsui  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
     26      1.1   tsutsui  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
     27      1.1   tsutsui  * THE POSSIBILITY OF SUCH DAMAGE.
     28      1.1   tsutsui  */
     29      1.1   tsutsui 
     30      1.1   tsutsui #ifdef ITECONSOLE
     31      1.1   tsutsui #include <sys/param.h>
     32      1.1   tsutsui 
     33      1.1   tsutsui #include <lib/libsa/stand.h>
     34      1.1   tsutsui 
     35      1.1   tsutsui #include <hp300/stand/common/samachdep.h>
     36      1.1   tsutsui #include <hp300/stand/common/itevar.h>
     37      1.1   tsutsui 
     38  1.2.6.1  perseant #include <hp300/dev/sti_diovar.h>
     39      1.1   tsutsui #include <hp300/dev/dioreg.h>
     40      1.1   tsutsui #include <hp300/dev/sgcreg.h>
     41      1.1   tsutsui #include <dev/ic/stireg.h>
     42      1.1   tsutsui 
     43      1.1   tsutsui /*
     44      1.1   tsutsui  * sti-specific data not available in the ite_data structure.
     45      1.1   tsutsui  * Since we will only configure one sti display, it is ok to use a global.
     46      1.1   tsutsui  */
     47      1.1   tsutsui static struct {
     48      1.1   tsutsui 	uint32_t	codeptr[STI_CODECNT];
     49      1.1   tsutsui 	uint8_t		*code;
     50      1.1   tsutsui 	uint32_t	fontbase;
     51      1.1   tsutsui 	u_int		firstchar, lastchar;
     52      1.1   tsutsui 	struct sti_cfg	cfg;
     53      1.1   tsutsui 	struct sti_ecfg	ecfg;
     54      1.1   tsutsui } sti;
     55      1.1   tsutsui 
     56      1.1   tsutsui #define parseshort1(addr, ofs) \
     57      1.1   tsutsui 	(((addr)[(ofs) +  3] << 8) | ((addr)[(ofs) +  7]))
     58      1.1   tsutsui #define parseword1(addr, ofs) \
     59      1.1   tsutsui 	(((addr)[(ofs) +  3] << 24) | ((addr)[(ofs) +  7] << 16) | \
     60      1.1   tsutsui 	 ((addr)[(ofs) + 11] <<  8) | ((addr)[(ofs) + 15]))
     61      1.1   tsutsui 
     62  1.2.6.1  perseant static void	sti_do_cursor(struct ite_data *);
     63  1.2.6.1  perseant static void	sti_fontinfo(struct ite_data *);
     64  1.2.6.1  perseant static void	sti_init(int);
     65  1.2.6.1  perseant static void	sti_inqcfg(struct sti_inqconfout *);
     66  1.2.6.1  perseant static void	sti_iteinit_common(struct ite_data *);
     67      1.1   tsutsui 
     68      1.1   tsutsui /* kinda similar to sti_dio_probe() */
     69      1.1   tsutsui int
     70      1.1   tsutsui sti_dio_probe(struct ite_data *ip)
     71      1.1   tsutsui {
     72      1.1   tsutsui 	int scode = ip->scode;
     73      1.1   tsutsui 	uint8_t *id_reg;
     74      1.1   tsutsui 
     75      1.1   tsutsui 	id_reg = (uint8_t *)sctoaddr(scode);
     76      1.1   tsutsui 	if (id_reg[DIOII_SIZEOFF] < STI_DIO_SIZE - 1)
     77      1.1   tsutsui 		return ENODEV;
     78      1.1   tsutsui 
     79      1.1   tsutsui 	id_reg = (uint8_t *)sctoaddr(scode + STI_DIO_SCODE_OFFSET);
     80      1.1   tsutsui 	if (id_reg[3] != STI_DEVTYPE1)
     81      1.1   tsutsui 		return ENODEV;
     82      1.1   tsutsui 
     83      1.1   tsutsui 	return 0;
     84      1.1   tsutsui }
     85      1.1   tsutsui 
     86      1.1   tsutsui void
     87      1.1   tsutsui sti_iteinit_dio(struct ite_data *ip)
     88      1.1   tsutsui {
     89      1.1   tsutsui 
     90  1.2.6.1  perseant 	ip->fbbase = (uint8_t *)sctoaddr(ip->scode + STI_DIO_SCODE_OFFSET);
     91      1.1   tsutsui 	sti_iteinit_common(ip);
     92      1.1   tsutsui }
     93      1.1   tsutsui 
     94      1.1   tsutsui void
     95      1.1   tsutsui sti_iteinit_sgc(struct ite_data *ip)
     96      1.1   tsutsui {
     97      1.1   tsutsui 
     98      1.1   tsutsui 	ip->fbbase = (uint8_t *)IIOV(SGC_BASE + (ip->scode * SGC_DEVSIZE));
     99      1.1   tsutsui 	sti_iteinit_common(ip);
    100      1.1   tsutsui }
    101      1.1   tsutsui 
    102      1.1   tsutsui /*
    103      1.1   tsutsui  * Initialize the sti device for ite's needs.
    104      1.1   tsutsui  * We don't bother to check for failures since
    105      1.1   tsutsui  * - we are in tight space already
    106      1.1   tsutsui  * - since romputchar() does not work with sti devices, there is no way we
    107      1.1   tsutsui  *   can report errors (although we could switch to serial...)
    108      1.1   tsutsui  */
    109  1.2.6.1  perseant static void
    110      1.1   tsutsui sti_iteinit_common(struct ite_data *ip)
    111      1.1   tsutsui {
    112      1.1   tsutsui 	int i;
    113      1.1   tsutsui 	size_t codesize, memsize;
    114      1.1   tsutsui 	uint8_t *va, *code;
    115      1.1   tsutsui 	u_int addr, eaddr, reglist, tmp;
    116      1.1   tsutsui 	struct sti_inqconfout cfg;
    117      1.1   tsutsui 	struct sti_einqconfout ecfg;
    118      1.1   tsutsui 
    119      1.1   tsutsui 	memset(&sti, 0, sizeof sti);
    120      1.1   tsutsui 	va = (uint8_t *)ip->fbbase;
    121      1.1   tsutsui 
    122      1.1   tsutsui 	/*
    123      1.1   tsutsui 	 * Read the microcode.
    124      1.1   tsutsui 	 */
    125      1.1   tsutsui 
    126      1.1   tsutsui 	for (i = 0; i < STI_CODECNT; i++)
    127      1.1   tsutsui 		sti.codeptr[i] =
    128      1.1   tsutsui 		    parseword1(va, (STI_CODEBASE_M68K << 2) + i * 0x10);
    129      1.1   tsutsui 
    130      1.1   tsutsui 	for (i = STI_END; sti.codeptr[i] == 0; i--)
    131      1.1   tsutsui 		continue;
    132      1.1   tsutsui 	codesize = sti.codeptr[i] - sti.codeptr[STI_BEGIN];
    133      1.1   tsutsui 	codesize = (codesize + 3) / 4;
    134      1.1   tsutsui 
    135      1.1   tsutsui 	sti.code = (uint8_t *)alloc(codesize);
    136      1.1   tsutsui 	code = sti.code;
    137      1.1   tsutsui 	addr = (u_int)va + sti.codeptr[STI_BEGIN];
    138      1.1   tsutsui 	eaddr = addr + codesize * 4;
    139      1.1   tsutsui 	for (; addr < eaddr; addr += 4)
    140      1.1   tsutsui 		*code++ = *(uint8_t *)addr;
    141      1.1   tsutsui 
    142      1.1   tsutsui 	for (i = STI_CODECNT - 1; i != 0; i--)
    143      1.1   tsutsui 		if (sti.codeptr[i] != 0) {
    144      1.1   tsutsui 			sti.codeptr[i] -= sti.codeptr[0];
    145      1.1   tsutsui 			sti.codeptr[i] /= 4;
    146      1.1   tsutsui 		}
    147      1.1   tsutsui 
    148      1.1   tsutsui 	sti.codeptr[0] = 0;
    149      1.1   tsutsui 	for (i = STI_END; sti.codeptr[i] == 0; i--);
    150      1.1   tsutsui 	sti.codeptr[i] = 0;
    151      1.1   tsutsui 
    152      1.1   tsutsui 	/*
    153      1.1   tsutsui 	 * Read the regions list.
    154      1.1   tsutsui 	 */
    155      1.1   tsutsui 
    156      1.1   tsutsui 	reglist = parseword1(va, 0x60);
    157      1.1   tsutsui 	for (i = 0; i < STI_REGION_MAX; i++) {
    158      1.1   tsutsui 		tmp = parseword1(va, (reglist & ~3) + i * 0x10);
    159      1.1   tsutsui 		sti.cfg.regions[i] = (u_int)va + ((tmp >> 18) << 12);
    160      1.1   tsutsui 		if (tmp & 0x4000)
    161      1.1   tsutsui 			break;
    162      1.1   tsutsui 	}
    163      1.1   tsutsui 
    164      1.1   tsutsui 	/*
    165      1.1   tsutsui 	 * Allocate scratch memory for the microcode if it needs it.
    166      1.1   tsutsui 	 */
    167      1.1   tsutsui 
    168      1.1   tsutsui 	sti.cfg.ext_cfg = &sti.ecfg;
    169      1.1   tsutsui 	memsize = parseword1(va, 0xa0);
    170      1.1   tsutsui 	if (memsize != 0)
    171      1.1   tsutsui 		sti.ecfg.addr = alloc(memsize);
    172      1.1   tsutsui 
    173      1.1   tsutsui 	/*
    174      1.1   tsutsui 	 * Initialize the display, and get geometry information.
    175      1.1   tsutsui 	 */
    176      1.1   tsutsui 
    177      1.1   tsutsui 	sti_init(0);
    178      1.1   tsutsui 
    179      1.1   tsutsui 	memset(&cfg, 0, sizeof cfg);
    180      1.1   tsutsui 	memset(&ecfg, 0, sizeof ecfg);
    181      1.1   tsutsui 	cfg.ext = &ecfg;
    182      1.1   tsutsui 	sti_inqcfg(&cfg);
    183      1.1   tsutsui 
    184      1.1   tsutsui 	if (cfg.owidth == cfg.width && cfg.oheight == cfg.height) {
    185      1.1   tsutsui 		sti.cfg.oscr_width = cfg.owidth = cfg.fbwidth - cfg.width;
    186      1.1   tsutsui 		sti.cfg.oscr_height = cfg.oheight = cfg.fbheight - cfg.height;
    187      1.1   tsutsui 	}
    188      1.1   tsutsui 
    189      1.1   tsutsui 	ip->dheight = cfg.height;
    190      1.1   tsutsui 	ip->dwidth = cfg.width;
    191      1.1   tsutsui 	ip->fbheight = cfg.fbheight;
    192      1.1   tsutsui 	ip->fbwidth = cfg.fbwidth;
    193      1.1   tsutsui 
    194      1.1   tsutsui 	/*
    195      1.1   tsutsui 	 * Get ready for ite operation!
    196      1.1   tsutsui 	 */
    197      1.1   tsutsui 
    198      1.1   tsutsui 	sti_init(1);
    199      1.1   tsutsui 	sti_fontinfo(ip);
    200      1.1   tsutsui 	sti_clear(ip, 0, 0, ip->rows, ip->cols);	/* necessary? */
    201      1.1   tsutsui }
    202      1.1   tsutsui 
    203      1.1   tsutsui void
    204      1.1   tsutsui sti_putc(struct ite_data *ip, int c, int dy, int dx)
    205      1.1   tsutsui {
    206      1.1   tsutsui 	sti_unpmv_t unpmv;
    207      1.1   tsutsui 	struct {
    208      1.1   tsutsui 		struct sti_unpmvflags flags;
    209      1.1   tsutsui 		struct sti_unpmvin in;
    210      1.1   tsutsui 		struct sti_unpmvout out;
    211      1.1   tsutsui 	} a;
    212      1.1   tsutsui 
    213      1.1   tsutsui 	memset(&a, 0, sizeof a);
    214      1.1   tsutsui 	a.flags.flags = STI_UNPMVF_WAIT;
    215      1.1   tsutsui 	a.in.bg_colour = STI_COLOUR_BLACK;
    216      1.1   tsutsui 	a.in.fg_colour = STI_COLOUR_WHITE;
    217      1.1   tsutsui 	a.in.x = dx * ip->ftwidth;
    218      1.1   tsutsui 	a.in.y = dy * ip->ftheight;
    219      1.1   tsutsui 	a.in.font_addr = (uint32_t *)((uint8_t *)ip->fbbase + sti.fontbase);
    220      1.1   tsutsui 	a.in.index = c;
    221      1.1   tsutsui 
    222      1.1   tsutsui 	unpmv = (sti_unpmv_t)(sti.code + sti.codeptr[STI_FONT_UNPMV]);
    223      1.1   tsutsui 	(*unpmv)(&a.flags, &a.in, &a.out, &sti.cfg);
    224      1.1   tsutsui }
    225      1.1   tsutsui 
    226      1.1   tsutsui void
    227      1.1   tsutsui sti_cursor(struct ite_data *ip, int flag)
    228      1.1   tsutsui {
    229      1.1   tsutsui 	switch (flag) {
    230      1.1   tsutsui 	case MOVE_CURSOR:
    231      1.1   tsutsui 		sti_do_cursor(ip);
    232      1.1   tsutsui 		/* FALLTHROUGH */
    233      1.1   tsutsui 	case DRAW_CURSOR:
    234      1.1   tsutsui 		ip->cursorx = ip->curx;
    235      1.1   tsutsui 		ip->cursory = ip->cury;
    236      1.1   tsutsui 		/* FALLTHROUGH */
    237  1.2.6.1  perseant 	case ERASE_CURSOR:
    238      1.1   tsutsui 	default:
    239      1.1   tsutsui 		sti_do_cursor(ip);
    240      1.1   tsutsui 		break;
    241      1.1   tsutsui 	}
    242      1.1   tsutsui }
    243      1.1   tsutsui 
    244  1.2.6.1  perseant static void
    245      1.1   tsutsui sti_do_cursor(struct ite_data *ip)
    246      1.1   tsutsui {
    247      1.1   tsutsui 	sti_blkmv_t blkmv;
    248      1.1   tsutsui 	struct {
    249      1.1   tsutsui 		struct sti_blkmvflags flags;
    250      1.1   tsutsui 		struct sti_blkmvin in;
    251      1.1   tsutsui 		struct sti_blkmvout out;
    252      1.1   tsutsui 	} a;
    253      1.1   tsutsui 
    254      1.1   tsutsui 	memset(&a, 0, sizeof a);
    255      1.1   tsutsui 	a.flags.flags = STI_BLKMVF_WAIT | STI_BLKMVF_COLR;
    256      1.1   tsutsui 	a.in.fg_colour = STI_COLOUR_BLACK;
    257      1.1   tsutsui 	a.in.bg_colour = STI_COLOUR_WHITE;
    258      1.1   tsutsui 	a.in.dstx = a.in.srcx = ip->cursorx * ip->ftwidth;
    259      1.1   tsutsui 	a.in.dsty = a.in.srcy = ip->cursory * ip->ftheight;
    260      1.1   tsutsui 	a.in.width = ip->ftwidth;
    261      1.1   tsutsui 	a.in.height = ip->ftheight;
    262      1.1   tsutsui 
    263      1.1   tsutsui 	blkmv = (sti_blkmv_t)(sti.code + sti.codeptr[STI_BLOCK_MOVE]);
    264      1.1   tsutsui 	(*blkmv)(&a.flags, &a.in, &a.out, &sti.cfg);
    265      1.1   tsutsui }
    266      1.1   tsutsui 
    267      1.1   tsutsui void
    268      1.1   tsutsui sti_clear(struct ite_data *ip, int sy, int sx, int h, int w)
    269      1.1   tsutsui {
    270      1.1   tsutsui 	sti_blkmv_t blkmv;
    271      1.1   tsutsui 	struct {
    272      1.1   tsutsui 		struct sti_blkmvflags flags;
    273      1.1   tsutsui 		struct sti_blkmvin in;
    274      1.1   tsutsui 		struct sti_blkmvout out;
    275      1.1   tsutsui 	} a;
    276      1.1   tsutsui 
    277      1.1   tsutsui 	memset(&a, 0, sizeof a);
    278      1.1   tsutsui 	a.flags.flags = STI_BLKMVF_WAIT | STI_BLKMVF_CLR;
    279      1.1   tsutsui 	a.in.bg_colour = STI_COLOUR_BLACK;
    280      1.1   tsutsui 	a.in.dstx = a.in.srcx = sx * ip->ftwidth;
    281      1.1   tsutsui 	a.in.dsty = a.in.srcy = sy * ip->ftheight;
    282      1.1   tsutsui 	a.in.width = w * ip->ftwidth;
    283      1.1   tsutsui 	a.in.height = h * ip->ftheight;
    284      1.1   tsutsui 
    285      1.1   tsutsui 	blkmv = (sti_blkmv_t)(sti.code + sti.codeptr[STI_BLOCK_MOVE]);
    286      1.1   tsutsui 	(*blkmv)(&a.flags, &a.in, &a.out, &sti.cfg);
    287      1.1   tsutsui }
    288      1.1   tsutsui 
    289      1.1   tsutsui void
    290      1.1   tsutsui sti_scroll(struct ite_data *ip)
    291      1.1   tsutsui {
    292      1.1   tsutsui 	sti_blkmv_t blkmv;
    293      1.1   tsutsui 	struct {
    294      1.1   tsutsui 		struct sti_blkmvflags flags;
    295      1.1   tsutsui 		struct sti_blkmvin in;
    296      1.1   tsutsui 		struct sti_blkmvout out;
    297      1.1   tsutsui 	} a;
    298      1.1   tsutsui 
    299      1.1   tsutsui 	memset(&a, 0, sizeof a);
    300      1.1   tsutsui 	a.flags.flags = STI_BLKMVF_WAIT;
    301      1.1   tsutsui 	a.in.bg_colour = STI_COLOUR_BLACK;
    302      1.1   tsutsui 	a.in.fg_colour = STI_COLOUR_WHITE;
    303      1.1   tsutsui 	a.in.dstx = a.in.srcx = 0;
    304      1.1   tsutsui 	a.in.dsty = 0;
    305      1.1   tsutsui 	a.in.srcy = ip->ftheight;
    306      1.1   tsutsui 	a.in.width = ip->dwidth;
    307      1.1   tsutsui 	a.in.height = (ip->rows - 1) * ip->ftheight;
    308      1.1   tsutsui 
    309      1.1   tsutsui 	blkmv = (sti_blkmv_t)(sti.code + sti.codeptr[STI_BLOCK_MOVE]);
    310      1.1   tsutsui 	(*blkmv)(&a.flags, &a.in, &a.out, &sti.cfg);
    311      1.1   tsutsui }
    312      1.1   tsutsui 
    313  1.2.6.1  perseant static void
    314      1.1   tsutsui sti_fontinfo(struct ite_data *ip)
    315      1.1   tsutsui {
    316      1.1   tsutsui 	uint32_t fontbase;
    317      1.1   tsutsui 	volatile uint8_t *fbbase = ip->fbbase;
    318      1.1   tsutsui 
    319      1.1   tsutsui 	fontbase = sti.fontbase = parseword1(fbbase, 0x30) & ~3;
    320      1.1   tsutsui 	ip->ftwidth = (uint8_t)fbbase[fontbase + 0x13];
    321      1.1   tsutsui 	ip->ftheight = (uint8_t)fbbase[fontbase + 0x17];
    322      1.1   tsutsui 	ip->rows = ip->dheight / ip->ftheight;
    323      1.1   tsutsui 	ip->cols = ip->dwidth / ip->ftwidth;
    324      1.1   tsutsui }
    325      1.1   tsutsui 
    326  1.2.6.1  perseant static void
    327      1.1   tsutsui sti_init(int full)
    328      1.1   tsutsui {
    329      1.1   tsutsui 	sti_init_t init;
    330      1.1   tsutsui 	struct {
    331      1.1   tsutsui 		struct sti_initflags flags;
    332      1.1   tsutsui 		struct sti_initin in;
    333      1.1   tsutsui 		struct sti_initout out;
    334      1.1   tsutsui 	} a;
    335      1.1   tsutsui 
    336      1.1   tsutsui 	memset(&a, 0, sizeof a);
    337      1.1   tsutsui 	a.flags.flags = STI_INITF_WAIT | STI_INITF_CMB | STI_INITF_EBET;
    338      1.1   tsutsui 	if (full)
    339      1.1   tsutsui 		a.flags.flags |= STI_INITF_TEXT | STI_INITF_PBET |
    340      1.1   tsutsui 		    STI_INITF_PBETI | STI_INITF_ICMT;
    341      1.1   tsutsui 	a.in.text_planes = 1;
    342      1.1   tsutsui 
    343      1.1   tsutsui 	init = (sti_init_t)(sti.code + sti.codeptr[STI_INIT_GRAPH]);
    344      1.1   tsutsui 	(*init)(&a.flags, &a.in, &a.out, &sti.cfg);
    345      1.1   tsutsui }
    346      1.1   tsutsui 
    347  1.2.6.1  perseant static void
    348      1.1   tsutsui sti_inqcfg(struct sti_inqconfout *ico)
    349      1.1   tsutsui {
    350      1.1   tsutsui 	sti_inqconf_t inqconf;
    351      1.1   tsutsui 	struct {
    352      1.1   tsutsui 		struct sti_inqconfflags flags;
    353      1.1   tsutsui 		struct sti_inqconfin in;
    354      1.1   tsutsui 	} a;
    355      1.1   tsutsui 
    356      1.1   tsutsui 	memset(&a, 0, sizeof a);
    357      1.1   tsutsui 	a.flags.flags = STI_INQCONFF_WAIT;
    358      1.1   tsutsui 
    359      1.1   tsutsui 	inqconf = (sti_inqconf_t)(sti.code + sti.codeptr[STI_INQ_CONF]);
    360      1.1   tsutsui 	(*inqconf)(&a.flags, &a.in, ico, &sti.cfg);
    361      1.1   tsutsui }
    362      1.1   tsutsui 
    363      1.1   tsutsui #endif
    364