1 1.6 tsutsui /* $NetBSD: ite_sti.c,v 1.6 2025/05/29 14:50:45 tsutsui 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.3 tsutsui #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.5 tsutsui static void sti_do_cursor(struct ite_data *); 63 1.5 tsutsui static void sti_fontinfo(struct ite_data *); 64 1.5 tsutsui static void sti_init(int); 65 1.5 tsutsui static void sti_inqcfg(struct sti_inqconfout *); 66 1.5 tsutsui 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.3 tsutsui 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.5 tsutsui 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.6 tsutsui 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.5 tsutsui 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.5 tsutsui 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.5 tsutsui 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.5 tsutsui 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