stifb.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:38k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * linux/drivers/video/sti/stifb.c - 
  3.  * Frame buffer driver for HP workstations with STI (standard text interface) 
  4.  * video firmware.
  5.  *
  6.  * Copyright (C) 2001 Helge Deller <deller@gmx.de>
  7.  * Portions Copyright (C) 2001 Thomas Bogendoerfer <tsbogend@alpha.franken.de>
  8.  * 
  9.  * Based on:
  10.  * - linux/drivers/video/artistfb.c -- Artist frame buffer driver
  11.  * Copyright (C) 2000 Philipp Rumpf <prumpf@tux.org>
  12.  *   - based on skeletonfb, which was
  13.  * Created 28 Dec 1997 by Geert Uytterhoeven
  14.  * - HP Xhp cfb-based X11 window driver for XFree86
  15.  * (c)Copyright 1992 Hewlett-Packard Co.
  16.  *
  17.  * 
  18.  *  The following graphics display devices (NGLE family) are supported by this driver:
  19.  *
  20.  *  HPA4070A known as "HCRX", a 1280x1024 color device with 8 planes
  21.  *  HPA4071A known as "HCRX24", a 1280x1024 color device with 24 planes,
  22.  * optionally available with a hardware accelerator as HPA4071A_Z
  23.  *  HPA1659A known as "CRX", a 1280x1024 color device with 8 planes
  24.  *  HPA1439A known as "CRX24", a 1280x1024 color device with 24 planes,
  25.  * optionally available with a hardware accelerator.
  26.  *  HPA1924A known as "GRX", a 1280x1024 grayscale device with 8 planes
  27.  *  HPA2269A known as "Dual CRX", a 1280x1024 color device with 8 planes,
  28.  * implements support for two displays on a single graphics card.
  29.  *  HP710C internal graphics support optionally available on the HP9000s710 SPU,
  30.  * supports 1280x1024 color displays with 8 planes.
  31.  *  HP710G same as HP710C, 1280x1024 grayscale only
  32.  *  HP710L same as HP710C, 1024x768 color only
  33.  *  HP712 internal graphics support on HP9000s712 SPU, supports 640x480, 
  34.  * 1024x768 or 1280x1024 color displays on 8 planes (Artist)
  35.  *
  36.  * This file is subject to the terms and conditions of the GNU General Public
  37.  * License.  See the file COPYING in the main directory of this archive
  38.  * for more details.
  39.  */
  40. /* TODO:
  41.  * - Artist gfx is the only supported chip atm,
  42.  * - remove the static fb_info to support multiple cards
  43.  * - remove the completely untested 1bpp mode
  44.  * - add support for h/w acceleration
  45.  * - add hardware cursor
  46.  * -
  47.  */
  48. /* on supported graphic devices you may:
  49.  * #define FALLBACK_TO_1BPP to fall back to 1 bpp, or
  50.  * #undef  FALLBACK_TO_1BPP to reject support for unsupported cards */
  51. #undef FALLBACK_TO_1BPP
  52. #include <linux/config.h>
  53. #include <linux/module.h>
  54. #include <linux/kernel.h>
  55. #include <linux/errno.h>
  56. #include <linux/string.h>
  57. #include <linux/mm.h>
  58. #include <linux/tty.h>
  59. #include <linux/slab.h>
  60. #include <linux/delay.h>
  61. #include <linux/fb.h>
  62. #include <linux/init.h>
  63. #include <linux/selection.h>
  64. #include <linux/ioport.h>
  65. #include <linux/pci.h>
  66. #include <video/fbcon.h>
  67. #include <video/fbcon-cfb8.h>
  68. #include <video/fbcon-cfb32.h>
  69. #include <asm/grfioctl.h> /* for HP-UX compatibility */
  70. #include "sticore.h"
  71. extern struct display_switch fbcon_sti; /* fbcon-sti.c */
  72. #ifdef __LP64__
  73. /* return virtual address */
  74. #define REGION_BASE(fb_info, index) 
  75. (fb_info->sti->glob_cfg->region_ptrs[index] | 0xffffffff00000000)
  76. #else
  77. /* return virtual address */
  78. #define REGION_BASE(fb_info, index) 
  79. fb_info->sti->glob_cfg->region_ptrs[index]
  80. #endif
  81. #define NGLEDEVDEPROM_CRT_REGION 1
  82. typedef struct {
  83. __s32 video_config_reg;
  84. __s32 misc_video_start;
  85. __s32 horiz_timing_fmt;
  86. __s32 serr_timing_fmt;
  87. __s32 vert_timing_fmt;
  88. __s32 horiz_state;
  89. __s32 vert_state;
  90. __s32 vtg_state_elements;
  91. __s32 pipeline_delay;
  92. __s32 misc_video_end;
  93. } video_setup_t;
  94. typedef struct {                  
  95. __s16 sizeof_ngle_data;
  96. __s16 x_size_visible;     /* visible screen dim in pixels  */
  97. __s16 y_size_visible;
  98. __s16 pad2[15];
  99. __s16 cursor_pipeline_delay;
  100. __s16 video_interleaves;
  101. __s32 pad3[11];
  102. } ngle_rom_t;
  103. struct stifb_info {
  104. struct fb_info info;
  105.         struct fb_var_screeninfo var;
  106.         struct fb_fix_screeninfo fix;
  107. struct display disp;
  108. struct sti_struct *sti;
  109. unsigned int id, real_id;
  110. int currcon;
  111. int cmap_reload:1;   
  112. int deviceSpecificConfig;
  113. ngle_rom_t ngle_rom;
  114. struct { u8 red, green, blue; } palette[256];
  115. #ifdef FBCON_HAS_CFB32
  116. union {
  117. u32 cfb32[16];
  118. } fbcon_cmap;
  119. #endif
  120. };
  121. static int stifb_force_bpp[MAX_STI_ROMS] = {0, };
  122. /* ------------------- chipset specific functions -------------------------- */
  123. /* offsets to graphic-chip internal registers */
  124. #define REG_1 0x000118
  125. #define REG_2 0x000480
  126. #define REG_3 0x0004a0
  127. #define REG_4 0x000600
  128. #define REG_6 0x000800
  129. #define REG_8 0x000820
  130. #define REG_9 0x000a04
  131. #define REG_10 0x018000
  132. #define REG_11 0x018004
  133. #define REG_12 0x01800c
  134. #define REG_13 0x018018
  135. #define REG_14   0x01801c
  136. #define REG_15 0x200000
  137. #define REG_15b0 0x200000
  138. #define REG_16b1 0x200005
  139. #define REG_16b3 0x200007
  140. #define REG_21 0x200218
  141. #define REG_22 0x0005a0
  142. #define REG_23 0x0005c0
  143. #define REG_26 0x200118
  144. #define REG_27 0x200308
  145. #define REG_32 0x21003c
  146. #define REG_33 0x210040
  147. #define REG_34 0x200008
  148. #define REG_35 0x018010
  149. #define REG_38 0x210020
  150. #define REG_39 0x210120
  151. #define REG_40 0x210130
  152. #define REG_42 0x210028
  153. #define REG_43 0x21002c
  154. #define REG_44 0x210030
  155. #define REG_45 0x210034
  156. #define READ_BYTE(fb,reg) __raw_readb((fb)->fix.mmio_start + (reg))
  157. #define READ_WORD(fb,reg) __raw_readl((fb)->fix.mmio_start + (reg))
  158. #define WRITE_BYTE(value,fb,reg) __raw_writeb((value),(fb)->fix.mmio_start + (reg))
  159. #define WRITE_WORD(value,fb,reg) __raw_writel((value),(fb)->fix.mmio_start + (reg))
  160. #define ENABLE 1 /* for enabling/disabling screen */
  161. #define DISABLE 0
  162. #define NGLE_LOCK(fb_info) do { } while (0) 
  163. #define NGLE_UNLOCK(fb_info) do { } while (0)
  164. static void
  165. SETUP_HW(struct stifb_info *fb)
  166. {
  167. char stat;
  168. do {
  169. stat = READ_BYTE(fb, REG_15b0);
  170. if (!stat)
  171.      stat = READ_BYTE(fb, REG_15b0);
  172. } while (stat);
  173. }
  174. static void
  175. SETUP_FB(struct stifb_info *fb)
  176. {
  177. unsigned int reg10_value = 0;
  178. SETUP_HW(fb);
  179. switch (fb->id)
  180. {
  181. case CRT_ID_VISUALIZE_EG:
  182. case S9000_ID_ARTIST:
  183. case S9000_ID_A1659A:
  184. reg10_value = 0x13601000;
  185. break;
  186. case S9000_ID_A1439A:
  187. if (fb->var.bits_per_pixel == 32)
  188. reg10_value = 0xBBA0A000;
  189. else 
  190. reg10_value = 0x13601000;
  191. break;
  192. case S9000_ID_HCRX:
  193. if (fb->var.bits_per_pixel == 32)
  194. reg10_value = 0xBBA0A000;
  195. else
  196. reg10_value = 0x13602000;
  197. break;
  198. case S9000_ID_TIMBER:
  199. case CRX24_OVERLAY_PLANES:
  200. reg10_value = 0x13602000;
  201. break;
  202. }
  203. if (reg10_value)
  204. WRITE_WORD(reg10_value, fb, REG_10);
  205. WRITE_WORD(0x83000300, fb, REG_14);
  206. SETUP_HW(fb);
  207. WRITE_BYTE(1, fb, REG_16b1);
  208. }
  209. static void
  210. START_IMAGE_COLORMAP_ACCESS(struct stifb_info *fb)
  211. {
  212. SETUP_HW(fb);
  213. WRITE_WORD(0xBBE0F000, fb, REG_10);
  214. WRITE_WORD(0x03000300, fb, REG_14);
  215. WRITE_WORD(~0, fb, REG_13);
  216. }
  217. static void
  218. WRITE_IMAGE_COLOR(struct stifb_info *fb, int index, int color) 
  219. {
  220. SETUP_HW(fb);
  221. WRITE_WORD(((0x100+index)<<2), fb, REG_3);
  222. WRITE_WORD(color, fb, REG_4);
  223. }
  224. static void
  225. FINISH_IMAGE_COLORMAP_ACCESS(struct stifb_info *fb) 
  226. {
  227. WRITE_WORD(0x400, fb, REG_2);
  228. if (fb->var.bits_per_pixel == 32) {
  229. WRITE_WORD(0x83000100, fb, REG_1);
  230. } else {
  231. if (fb->id == S9000_ID_ARTIST || fb->id == CRT_ID_VISUALIZE_EG)
  232. WRITE_WORD(0x80000100, fb, REG_26);
  233. else
  234. WRITE_WORD(0x80000100, fb, REG_1);
  235. }
  236. SETUP_FB(fb);
  237. }
  238. static void
  239. SETUP_RAMDAC(struct stifb_info *fb) 
  240. {
  241. SETUP_HW(fb);
  242. WRITE_WORD(0x04000000, fb, 0x1020);
  243. WRITE_WORD(0xff000000, fb, 0x1028);
  244. }
  245. static void 
  246. CRX24_SETUP_RAMDAC(struct stifb_info *fb) 
  247. {
  248. SETUP_HW(fb);
  249. WRITE_WORD(0x04000000, fb, 0x1000);
  250. WRITE_WORD(0x02000000, fb, 0x1004);
  251. WRITE_WORD(0xff000000, fb, 0x1008);
  252. WRITE_WORD(0x05000000, fb, 0x1000);
  253. WRITE_WORD(0x02000000, fb, 0x1004);
  254. WRITE_WORD(0x03000000, fb, 0x1008);
  255. }
  256. #if 0
  257. static void 
  258. HCRX_SETUP_RAMDAC(struct stifb_info *fb)
  259. {
  260. WRITE_WORD(0xffffffff, fb, REG_32);
  261. }
  262. #endif
  263. static void 
  264. CRX24_SET_OVLY_MASK(struct stifb_info *fb)
  265. {
  266. SETUP_HW(fb);
  267. WRITE_WORD(0x13a02000, fb, REG_11);
  268. WRITE_WORD(0x03000300, fb, REG_14);
  269. WRITE_WORD(0x000017f0, fb, REG_3);
  270. WRITE_WORD(0xffffffff, fb, REG_13);
  271. WRITE_WORD(0xffffffff, fb, REG_22);
  272. WRITE_WORD(0x00000000, fb, REG_23);
  273. }
  274. static void
  275. ENABLE_DISABLE_DISPLAY(struct stifb_info *fb, int enable)
  276. {
  277. unsigned int value = enable ? 0x43000000 : 0x03000000;
  278.         SETUP_HW(fb);
  279.         WRITE_WORD(0x06000000, fb, 0x1030);
  280.         WRITE_WORD(value,  fb, 0x1038);
  281. }
  282. static void 
  283. CRX24_ENABLE_DISABLE_DISPLAY(struct stifb_info *fb, int enable)
  284. {
  285. unsigned int value = enable ? 0x10000000 : 0x30000000;
  286. SETUP_HW(fb);
  287. WRITE_WORD(0x01000000, fb, 0x1000);
  288. WRITE_WORD(0x02000000, fb, 0x1004);
  289. WRITE_WORD(value, fb, 0x1008);
  290. }
  291. static void
  292. ARTIST_ENABLE_DISABLE_DISPLAY(struct stifb_info *fb, int enable) 
  293. {
  294. u32 DregsMiscVideo = REG_21;
  295. u32 DregsMiscCtl = REG_27;
  296. SETUP_HW(fb);
  297. if (enable) {
  298.   WRITE_WORD(READ_WORD(fb, DregsMiscVideo) | 0x0A000000, fb, DregsMiscVideo);
  299.   WRITE_WORD(READ_WORD(fb, DregsMiscCtl)   | 0x00800000, fb, DregsMiscCtl);
  300. } else {
  301.   WRITE_WORD(READ_WORD(fb, DregsMiscVideo) & ~0x0A000000, fb, DregsMiscVideo);
  302.   WRITE_WORD(READ_WORD(fb, DregsMiscCtl)   & ~0x00800000, fb, DregsMiscCtl);
  303. }
  304. }
  305. #define GET_ROMTABLE_INDEX(fb) 
  306. (READ_BYTE(fb, REG_16b3) - 1)
  307. #define HYPER_CONFIG_PLANES_24 0x00000100
  308. #define IS_24_DEVICE(fb) 
  309. (fb->deviceSpecificConfig & HYPER_CONFIG_PLANES_24)
  310. #define IS_888_DEVICE(fb) 
  311. (!(IS_24_DEVICE(fb)))
  312. #define GET_FIFO_SLOTS(fb, cnt, numslots)
  313. { while (cnt < numslots) 
  314. cnt = READ_WORD(fb, REG_34);
  315. cnt -= numslots;
  316. }
  317. #define     IndexedDcd 0 /* Pixel data is indexed (pseudo) color */
  318. #define     Otc04 2 /* Pixels in each longword transfer (4) */
  319. #define     Otc32 5 /* Pixels in each longword transfer (32) */
  320. #define     Ots08 3 /* Each pixel is size (8)d transfer (1) */
  321. #define     OtsIndirect 6 /* Each bit goes through FG/BG color(8) */
  322. #define     AddrLong 5 /* FB address is Long aligned (pixel) */
  323. #define     BINovly 0x2 /* 8 bit overlay */
  324. #define     BINapp0I 0x0 /* Application Buffer 0, Indexed */
  325. #define     BINapp1I 0x1 /* Application Buffer 1, Indexed */
  326. #define     BINapp0F8 0xa /* Application Buffer 0, Fractional 8-8-8 */
  327. #define     BINattr 0xd /* Attribute Bitmap */
  328. #define     RopSrc  0x3
  329. #define     BitmapExtent08  3 /* Each write hits ( 8) bits in depth */
  330. #define     BitmapExtent32  5 /* Each write hits (32) bits in depth */
  331. #define     DataDynamic     0 /* Data register reloaded by direct access */
  332. #define     MaskDynamic     1 /* Mask register reloaded by direct access */
  333. #define     MaskOtc     0 /* Mask contains Object Count valid bits */
  334. #define MaskAddrOffset(offset) (offset)
  335. #define StaticReg(en) (en)
  336. #define BGx(en) (en)
  337. #define FGx(en) (en)
  338. #define BAJustPoint(offset) (offset)
  339. #define BAIndexBase(base) (base)
  340. #define BA(F,C,S,A,J,B,I) 
  341. (((F)<<31)|((C)<<27)|((S)<<24)|((A)<<21)|((J)<<16)|((B)<<12)|(I))
  342. #define IBOvals(R,M,X,S,D,L,B,F) 
  343. (((R)<<8)|((M)<<16)|((X)<<24)|((S)<<29)|((D)<<28)|((L)<<31)|((B)<<1)|(F))
  344. #define NGLE_QUICK_SET_IMAGE_BITMAP_OP(fb, val) 
  345. WRITE_WORD(val, fb, REG_14)
  346. #define NGLE_QUICK_SET_DST_BM_ACCESS(fb, val) 
  347. WRITE_WORD(val, fb, REG_11)
  348. #define NGLE_QUICK_SET_CTL_PLN_REG(fb, val) 
  349. WRITE_WORD(val, fb, REG_12)
  350. #define NGLE_REALLY_SET_IMAGE_PLANEMASK(fb, plnmsk32) 
  351. WRITE_WORD(plnmsk32, fb, REG_13)
  352. #define NGLE_REALLY_SET_IMAGE_FG_COLOR(fb, fg32) 
  353. WRITE_WORD(fg32, fb, REG_35)
  354. #define NGLE_SET_TRANSFERDATA(fb, val) 
  355. WRITE_WORD(val, fb, REG_8)
  356. #define NGLE_SET_DSTXY(fb, val) 
  357. WRITE_WORD(val, fb, REG_6)
  358. #define NGLE_LONG_FB_ADDRESS(fbaddrbase, x, y) (
  359. (u32) (fbaddrbase) +
  360.     ( (unsigned int)  ( (y) << 13      ) |
  361. (unsigned int)  ( (x) << 2       ) )
  362. )
  363. #define NGLE_BINC_SET_DSTADDR(fb, addr) 
  364. WRITE_WORD(addr, fb, REG_3)
  365. #define NGLE_BINC_SET_SRCADDR(fb, addr) 
  366. WRITE_WORD(addr, fb, REG_2)
  367. #define NGLE_BINC_SET_DSTMASK(fb, mask) 
  368. WRITE_WORD(mask, fb, REG_22)
  369. #define NGLE_BINC_WRITE32(fb, data32) 
  370. WRITE_WORD(data32, fb, REG_23)
  371. #define START_COLORMAPLOAD(fb, cmapBltCtlData32) 
  372. WRITE_WORD((cmapBltCtlData32), fb, REG_38)
  373. #define SET_LENXY_START_RECFILL(fb, lenxy) 
  374. WRITE_WORD(lenxy, fb, REG_9)
  375. static void
  376. HYPER_ENABLE_DISABLE_DISPLAY(struct stifb_info *fb, int enable)
  377. {
  378. u32 DregsHypMiscVideo = REG_33;
  379. unsigned int value;
  380. SETUP_HW(fb);
  381. value = READ_WORD(fb, DregsHypMiscVideo);
  382. if (enable)
  383. value |= 0x0A000000;
  384. else
  385. value &= ~0x0A000000;
  386. WRITE_WORD(value, fb, DregsHypMiscVideo);
  387. }
  388. /* BufferNumbers used by SETUP_ATTR_ACCESS() */
  389. #define BUFF0_CMAP0 0x00001e02
  390. #define BUFF1_CMAP0 0x02001e02
  391. #define BUFF1_CMAP3 0x0c001e02
  392. #define ARTIST_CMAP0 0x00000102
  393. #define HYPER_CMAP8 0x00000100
  394. #define HYPER_CMAP24 0x00000800
  395. static void
  396. SETUP_ATTR_ACCESS(struct stifb_info *fb, unsigned BufferNumber)
  397. {
  398. SETUP_HW(fb);
  399. WRITE_WORD(0x2EA0D000, fb, REG_11);
  400. WRITE_WORD(0x23000302, fb, REG_14);
  401. WRITE_WORD(BufferNumber, fb, REG_12);
  402. WRITE_WORD(0xffffffff, fb, REG_8);
  403. }
  404. static void
  405. SET_ATTR_SIZE(struct stifb_info *fb, int width, int height) 
  406. {
  407. WRITE_WORD(0x00000000, fb, REG_6);
  408. WRITE_WORD((width<<16) | height, fb, REG_9);
  409. WRITE_WORD(0x05000000, fb, REG_6);
  410. WRITE_WORD(0x00040001, fb, REG_9);
  411. }
  412. static void
  413. FINISH_ATTR_ACCESS(struct stifb_info *fb) 
  414. {
  415. SETUP_HW(fb);
  416. WRITE_WORD(0x00000000, fb, REG_12);
  417. }
  418. static void
  419. elkSetupPlanes(struct stifb_info *fb)
  420. {
  421. SETUP_RAMDAC(fb);
  422. SETUP_FB(fb);
  423. }
  424. static void 
  425. ngleSetupAttrPlanes(struct stifb_info *fb, int BufferNumber)
  426. {
  427. SETUP_ATTR_ACCESS(fb, BufferNumber);
  428. SET_ATTR_SIZE(fb, fb->var.xres, fb->var.yres);
  429. FINISH_ATTR_ACCESS(fb);
  430. SETUP_FB(fb);
  431. }
  432. static void
  433. rattlerSetupPlanes(struct stifb_info *fb)
  434. {
  435. CRX24_SETUP_RAMDAC(fb);
  436.     
  437. /* replacement for: SETUP_FB(fb, CRX24_OVERLAY_PLANES); */
  438. WRITE_WORD(0x83000300, fb, REG_14);
  439. SETUP_HW(fb);
  440. WRITE_BYTE(1, fb, REG_16b1);
  441. /* XXX: replace by fb_setmem(), smem_start or screen_base ? */
  442. memset_io(fb->fix.smem_start, 0xff,
  443. fb->var.yres*fb->fix.line_length);
  444.     
  445. CRX24_SET_OVLY_MASK(fb);
  446. SETUP_FB(fb);
  447. }
  448. #define HYPER_CMAP_TYPE 0
  449. #define NGLE_CMAP_INDEXED0_TYPE 0
  450. #define NGLE_CMAP_OVERLAY_TYPE 3
  451. /* typedef of LUT (Colormap) BLT Control Register */
  452. typedef union /* Note assumption that fields are packed left-to-right */
  453. { u32 all;
  454. struct
  455. {
  456. unsigned enable              :  1;
  457. unsigned waitBlank           :  1;
  458. unsigned reserved1           :  4;
  459. unsigned lutOffset           : 10;   /* Within destination LUT */
  460. unsigned lutType             :  2;   /* Cursor, image, overlay */
  461. unsigned reserved2           :  4;
  462. unsigned length              : 10;
  463. } fields;
  464. } NgleLutBltCtl;
  465. #if 0
  466. static NgleLutBltCtl
  467. setNgleLutBltCtl(struct stifb_info *fb, int offsetWithinLut, int length)
  468. {
  469. NgleLutBltCtl lutBltCtl;
  470. /* set enable, zero reserved fields */
  471. lutBltCtl.all           = 0x80000000;
  472. lutBltCtl.fields.length = length;
  473. switch (fb->id) 
  474. {
  475. case S9000_ID_A1439A: /* CRX24 */
  476. if (fb->var.bits_per_pixel == 8) {
  477. lutBltCtl.fields.lutType = NGLE_CMAP_OVERLAY_TYPE;
  478. lutBltCtl.fields.lutOffset = 0;
  479. } else {
  480. lutBltCtl.fields.lutType = NGLE_CMAP_INDEXED0_TYPE;
  481. lutBltCtl.fields.lutOffset = 0 * 256;
  482. }
  483. break;
  484. case S9000_ID_ARTIST:
  485. lutBltCtl.fields.lutType = NGLE_CMAP_INDEXED0_TYPE;
  486. lutBltCtl.fields.lutOffset = 0 * 256;
  487. break;
  488. default:
  489. lutBltCtl.fields.lutType = NGLE_CMAP_INDEXED0_TYPE;
  490. lutBltCtl.fields.lutOffset = 0;
  491. break;
  492. }
  493. /* Offset points to start of LUT.  Adjust for within LUT */
  494. lutBltCtl.fields.lutOffset += offsetWithinLut;
  495. return lutBltCtl;
  496. }
  497. #endif
  498. static NgleLutBltCtl
  499. setHyperLutBltCtl(struct stifb_info *fb, int offsetWithinLut, int length) 
  500. {
  501. NgleLutBltCtl lutBltCtl;
  502. /* set enable, zero reserved fields */
  503. lutBltCtl.all = 0x80000000;
  504. lutBltCtl.fields.length = length;
  505. lutBltCtl.fields.lutType = HYPER_CMAP_TYPE;
  506. /* Expect lutIndex to be 0 or 1 for image cmaps, 2 or 3 for overlay cmaps */
  507. if (fb->var.bits_per_pixel == 8)
  508. lutBltCtl.fields.lutOffset = 2 * 256;
  509. else
  510. lutBltCtl.fields.lutOffset = 0 * 256;
  511. /* Offset points to start of LUT.  Adjust for within LUT */
  512. lutBltCtl.fields.lutOffset += offsetWithinLut;
  513. return lutBltCtl;
  514. }
  515. static void hyperUndoITE(struct stifb_info *fb)
  516. {
  517. int nFreeFifoSlots = 0;
  518. u32 fbAddr;
  519. NGLE_LOCK(fb);
  520. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 1);
  521. WRITE_WORD(0xffffffff, fb, REG_32);
  522. /* Write overlay transparency mask so only entry 255 is transparent */
  523. /* Hardware setup for full-depth write to "magic" location */
  524. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 7);
  525. NGLE_QUICK_SET_DST_BM_ACCESS(fb, 
  526. BA(IndexedDcd, Otc04, Ots08, AddrLong,
  527. BAJustPoint(0), BINovly, BAIndexBase(0)));
  528. NGLE_QUICK_SET_IMAGE_BITMAP_OP(fb,
  529. IBOvals(RopSrc, MaskAddrOffset(0),
  530. BitmapExtent08, StaticReg(0),
  531. DataDynamic, MaskOtc, BGx(0), FGx(0)));
  532. /* Now prepare to write to the "magic" location */
  533. fbAddr = NGLE_LONG_FB_ADDRESS(0, 1532, 0);
  534. NGLE_BINC_SET_DSTADDR(fb, fbAddr);
  535. NGLE_REALLY_SET_IMAGE_PLANEMASK(fb, 0xffffff);
  536. NGLE_BINC_SET_DSTMASK(fb, 0xffffffff);
  537. /* Finally, write a zero to clear the mask */
  538. NGLE_BINC_WRITE32(fb, 0);
  539. NGLE_UNLOCK(fb);
  540. }
  541. static void 
  542. ngleDepth8_ClearImagePlanes(struct stifb_info *fb)
  543. {
  544. /* FIXME! */
  545. }
  546. static void 
  547. ngleDepth24_ClearImagePlanes(struct stifb_info *fb)
  548. {
  549. /* FIXME! */
  550. }
  551. static void
  552. ngleResetAttrPlanes(struct stifb_info *fb, unsigned int ctlPlaneReg)
  553. {
  554. int nFreeFifoSlots = 0;
  555. u32 packed_dst;
  556. u32 packed_len;
  557. NGLE_LOCK(fb);
  558. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 4);
  559. NGLE_QUICK_SET_DST_BM_ACCESS(fb, 
  560.      BA(IndexedDcd, Otc32, OtsIndirect,
  561. AddrLong, BAJustPoint(0),
  562. BINattr, BAIndexBase(0)));
  563. NGLE_QUICK_SET_CTL_PLN_REG(fb, ctlPlaneReg);
  564. NGLE_SET_TRANSFERDATA(fb, 0xffffffff);
  565. NGLE_QUICK_SET_IMAGE_BITMAP_OP(fb,
  566.        IBOvals(RopSrc, MaskAddrOffset(0),
  567.        BitmapExtent08, StaticReg(1),
  568.        DataDynamic, MaskOtc,
  569.        BGx(0), FGx(0)));
  570. packed_dst = 0;
  571. packed_len = (fb->var.xres << 16) | fb->var.yres;
  572. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 2);
  573. NGLE_SET_DSTXY(fb, packed_dst);
  574. SET_LENXY_START_RECFILL(fb, packed_len);
  575. /*
  576.  * In order to work around an ELK hardware problem (Buffy doesn't
  577.  * always flush it's buffers when writing to the attribute
  578.  * planes), at least 4 pixels must be written to the attribute
  579.  * planes starting at (X == 1280) and (Y != to the last Y written
  580.  * by BIF):
  581.  */
  582. if (fb->id == S9000_ID_A1659A) {   /* ELK_DEVICE_ID */
  583. /* It's safe to use scanline zero: */
  584. packed_dst = (1280 << 16);
  585. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 2);
  586. NGLE_SET_DSTXY(fb, packed_dst);
  587. packed_len = (4 << 16) | 1;
  588. SET_LENXY_START_RECFILL(fb, packed_len);
  589. }   /* ELK Hardware Kludge */
  590. /**** Finally, set the Control Plane Register back to zero: ****/
  591. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 1);
  592. NGLE_QUICK_SET_CTL_PLN_REG(fb, 0);
  593. NGLE_UNLOCK(fb);
  594. }
  595.     
  596. static void
  597. ngleClearOverlayPlanes(struct stifb_info *fb, int mask, int data)
  598. {
  599. int nFreeFifoSlots = 0;
  600. u32 packed_dst;
  601. u32 packed_len;
  602.     
  603. NGLE_LOCK(fb);
  604. /* Hardware setup */
  605. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 8);
  606. NGLE_QUICK_SET_DST_BM_ACCESS(fb, 
  607.      BA(IndexedDcd, Otc04, Ots08, AddrLong,
  608. BAJustPoint(0), BINovly, BAIndexBase(0)));
  609.         NGLE_SET_TRANSFERDATA(fb, 0xffffffff);  /* Write foreground color */
  610.         NGLE_REALLY_SET_IMAGE_FG_COLOR(fb, data);
  611.         NGLE_REALLY_SET_IMAGE_PLANEMASK(fb, mask);
  612.     
  613.         packed_dst = 0;
  614.         packed_len = (fb->var.xres << 16) | fb->var.yres;
  615.         NGLE_SET_DSTXY(fb, packed_dst);
  616.     
  617.         /* Write zeroes to overlay planes */        
  618. NGLE_QUICK_SET_IMAGE_BITMAP_OP(fb,
  619.        IBOvals(RopSrc, MaskAddrOffset(0),
  620.        BitmapExtent08, StaticReg(0),
  621.        DataDynamic, MaskOtc, BGx(0), FGx(0)));
  622.        
  623.         SET_LENXY_START_RECFILL(fb, packed_len);
  624. NGLE_UNLOCK(fb);
  625. }
  626. static void 
  627. hyperResetPlanes(struct stifb_info *fb, int enable)
  628. {
  629. unsigned int controlPlaneReg;
  630. NGLE_LOCK(fb);
  631. if (IS_24_DEVICE(fb))
  632. if (fb->var.bits_per_pixel == 32)
  633. controlPlaneReg = 0x04000F00;
  634. else
  635. controlPlaneReg = 0x00000F00;   /* 0x00000800 should be enought, but lets clear all 4 bits */
  636. else
  637. controlPlaneReg = 0x00000F00; /* 0x00000100 should be enought, but lets clear all 4 bits */
  638. switch (enable) {
  639. case 1: /* ENABLE */
  640. /* clear screen */
  641. if (IS_24_DEVICE(fb))
  642. ngleDepth24_ClearImagePlanes(fb);
  643. else
  644. ngleDepth8_ClearImagePlanes(fb);
  645. /* Paint attribute planes for default case.
  646.  * On Hyperdrive, this means all windows using overlay cmap 0. */
  647. ngleResetAttrPlanes(fb, controlPlaneReg);
  648. /* clear overlay planes */
  649.         ngleClearOverlayPlanes(fb, 0xff, 255);
  650. /**************************************************
  651.  ** Also need to counteract ITE settings 
  652.  **************************************************/
  653. hyperUndoITE(fb);
  654. break;
  655. case 0: /* DISABLE */
  656. /* clear screen */
  657. if (IS_24_DEVICE(fb))
  658. ngleDepth24_ClearImagePlanes(fb);
  659. else
  660. ngleDepth8_ClearImagePlanes(fb);
  661. ngleResetAttrPlanes(fb, controlPlaneReg);
  662. ngleClearOverlayPlanes(fb, 0xff, 0);
  663. break;
  664. case -1: /* RESET */
  665. hyperUndoITE(fb);
  666. ngleResetAttrPlanes(fb, controlPlaneReg);
  667. break;
  668.      }
  669. NGLE_UNLOCK(fb);
  670. }
  671. /* Return pointer to in-memory structure holding ELK device-dependent ROM values. */
  672. static void 
  673. ngleGetDeviceRomData(struct stifb_info *fb)
  674. {
  675. #if 0
  676. XXX: FIXME: !!!
  677. int *pBytePerLongDevDepData;/* data byte == LSB */
  678. int  *pRomTable;
  679. NgleDevRomData *pPackedDevRomData;
  680. int sizePackedDevRomData = sizeof(*pPackedDevRomData);
  681. char *pCard8;
  682. int i;
  683. char *mapOrigin = NULL;
  684.     
  685. int romTableIdx;
  686. pPackedDevRomData = fb->ngle_rom;
  687. SETUP_HW(fb);
  688. if (fb->id == S9000_ID_ARTIST) {
  689. pPackedDevRomData->cursor_pipeline_delay = 4;
  690. pPackedDevRomData->video_interleaves     = 4;
  691. } else {
  692. /* Get pointer to unpacked byte/long data in ROM */
  693. pBytePerLongDevDepData = fb->sti->regions[NGLEDEVDEPROM_CRT_REGION];
  694. /* Tomcat supports several resolutions: 1280x1024, 1024x768, 640x480 */
  695. if (fb->id == S9000_ID_TOMCAT)
  696. {
  697.     /*  jump to the correct ROM table  */
  698.     GET_ROMTABLE_INDEX(romTableIdx);
  699.     while  (romTableIdx > 0)
  700.     {
  701. pCard8 = (Card8 *) pPackedDevRomData;
  702. pRomTable = pBytePerLongDevDepData;
  703. /* Pack every fourth byte from ROM into structure */
  704. for (i = 0; i < sizePackedDevRomData; i++)
  705. {
  706.     *pCard8++ = (Card8) (*pRomTable++);
  707. }
  708. pBytePerLongDevDepData = (Card32 *)
  709. ((Card8 *) pBytePerLongDevDepData +
  710.        pPackedDevRomData->sizeof_ngle_data);
  711. romTableIdx--;
  712.     }
  713. }
  714. pCard8 = (Card8 *) pPackedDevRomData;
  715. /* Pack every fourth byte from ROM into structure */
  716. for (i = 0; i < sizePackedDevRomData; i++)
  717. {
  718.     *pCard8++ = (Card8) (*pBytePerLongDevDepData++);
  719. }
  720.     }
  721.     SETUP_FB(fb);
  722. #endif
  723. }
  724. #define HYPERBOWL_MODE_FOR_8_OVER_88_LUT0_NO_TRANSPARENCIES 4
  725. #define HYPERBOWL_MODE01_8_24_LUT0_TRANSPARENT_LUT1_OPAQUE 8
  726. #define HYPERBOWL_MODE01_8_24_LUT0_OPAQUE_LUT1_OPAQUE 10
  727. #define HYPERBOWL_MODE2_8_24 15
  728. /* HCRX specific boot-time initialization */
  729. static void __init
  730. SETUP_HCRX(struct stifb_info *fb)
  731. {
  732. int hyperbowl;
  733.         int nFreeFifoSlots = 0;
  734. if (fb->id != S9000_ID_HCRX)
  735. return;
  736. /* Initialize Hyperbowl registers */
  737. GET_FIFO_SLOTS(fb, nFreeFifoSlots, 7);
  738. if (IS_24_DEVICE(fb)) {
  739. hyperbowl = (fb->var.bits_per_pixel == 32) ?
  740. HYPERBOWL_MODE01_8_24_LUT0_TRANSPARENT_LUT1_OPAQUE :
  741. HYPERBOWL_MODE01_8_24_LUT0_OPAQUE_LUT1_OPAQUE;
  742. /* First write to Hyperbowl must happen twice (bug) */
  743. WRITE_WORD(hyperbowl, fb, REG_40);
  744. WRITE_WORD(hyperbowl, fb, REG_40);
  745. WRITE_WORD(HYPERBOWL_MODE2_8_24, fb, REG_39);
  746. WRITE_WORD(0x014c0148, fb, REG_42); /* Set lut 0 to be the direct color */
  747. WRITE_WORD(0x404c4048, fb, REG_43);
  748. WRITE_WORD(0x034c0348, fb, REG_44);
  749. WRITE_WORD(0x444c4448, fb, REG_45);
  750. } else {
  751. hyperbowl = HYPERBOWL_MODE_FOR_8_OVER_88_LUT0_NO_TRANSPARENCIES;
  752. /* First write to Hyperbowl must happen twice (bug) */
  753. WRITE_WORD(hyperbowl, fb, REG_40);
  754. WRITE_WORD(hyperbowl, fb, REG_40);
  755. WRITE_WORD(0x00000000, fb, REG_42);
  756. WRITE_WORD(0x00000000, fb, REG_43);
  757. WRITE_WORD(0x00000000, fb, REG_44);
  758. WRITE_WORD(0x444c4048, fb, REG_45);
  759. }
  760. }
  761. /* ------------------- driver specific functions --------------------------- */
  762. static int
  763. stifb_getcolreg(u_int regno, u_int *red, u_int *green,
  764.       u_int *blue, u_int *transp, struct fb_info *info)
  765. {
  766. struct stifb_info *fb = (struct stifb_info *) info;
  767. if (regno > 255)
  768. return 1;
  769. *red = (fb->palette[regno].red<<8) | fb->palette[regno].red;
  770. *green = (fb->palette[regno].green<<8) | fb->palette[regno].green;
  771. *blue = (fb->palette[regno].blue<<8) | fb->palette[regno].blue;
  772. *transp = 0;
  773. return 0;
  774. }
  775. static int
  776. stifb_setcolreg(u_int regno, u_int red, u_int green,
  777.       u_int blue, u_int transp, struct fb_info *info)
  778. {
  779. struct stifb_info *fb = (struct stifb_info *) info;
  780. if (regno > 255)
  781. return 1;
  782. red >>= 8;
  783. green >>= 8;
  784. blue >>= 8;
  785. if ((fb->palette[regno].red != red) ||
  786.     (fb->palette[regno].green != green) ||
  787.     (fb->palette[regno].blue != blue))
  788. fb->cmap_reload = 1;
  789.     
  790. fb->palette[regno].red = red;
  791. fb->palette[regno].green = green;
  792. fb->palette[regno].blue = blue;
  793.     
  794. #ifdef FBCON_HAS_CFB32
  795. if (regno < 16 && fb->var.bits_per_pixel == 32) {
  796. fb->fbcon_cmap.cfb32[regno] = ((red << 16) |
  797.        (green << 8) |
  798.        (blue << 0) |
  799.        (transp << 24));
  800. }
  801. #endif
  802. return 0;
  803. }
  804. static void
  805. stifb_loadcmap(struct stifb_info *fb)
  806. {
  807. u32 color;
  808. int i;
  809. if (!fb->cmap_reload)
  810. return;
  811. START_IMAGE_COLORMAP_ACCESS(fb);
  812. for (i = 0; i < 256; i++) {
  813. if (fb->var.bits_per_pixel > 8) {
  814. color = (i << 16) | (i << 8) | i;
  815. } else {
  816. if (fb->var.grayscale) {
  817. /* gray = 0.30*R + 0.59*G + 0.11*B */
  818. color = ((fb->palette[i].red * 77) +
  819.  (fb->palette[i].green * 151) +
  820.  (fb->palette[i].blue * 28)) >> 8;
  821. } else {
  822. color = ((fb->palette[i].red << 16) |
  823.  (fb->palette[i].green << 8) |
  824.  (fb->palette[i].blue));
  825. }
  826. }
  827. WRITE_IMAGE_COLOR(fb, i, color);
  828. }
  829. if (fb->id == S9000_ID_HCRX) {
  830. NgleLutBltCtl lutBltCtl;
  831. lutBltCtl = setHyperLutBltCtl(fb,
  832. 0, /* Offset w/i LUT */
  833. 256); /* Load entire LUT */
  834. NGLE_BINC_SET_SRCADDR(fb,
  835. NGLE_LONG_FB_ADDRESS(0, 0x100, 0)); 
  836. /* 0x100 is same as used in WRITE_IMAGE_COLOR() */
  837. START_COLORMAPLOAD(fb, lutBltCtl.all);
  838. SETUP_FB(fb);
  839. } else {
  840. /* cleanup colormap hardware */
  841. FINISH_IMAGE_COLORMAP_ACCESS(fb);
  842. }
  843. fb->cmap_reload = 0;
  844. }
  845. static int
  846. stifb_get_fix(struct fb_fix_screeninfo *fix, int con,
  847.       struct fb_info *info)
  848. {
  849. memcpy (fix, &((struct stifb_info *)info)->fix, sizeof (*fix));
  850. return 0;
  851. }
  852. static int
  853. stifb_get_var(struct fb_var_screeninfo *var, int con,
  854.       struct fb_info *info)
  855. {
  856. memcpy (var, &((struct stifb_info *)info)->var, sizeof (*var));
  857. return 0;
  858. }
  859. static int
  860. stifb_set_var(struct fb_var_screeninfo *var, int con,
  861.       struct fb_info *info)
  862. {
  863. struct display *disp;
  864. if (con >= 0)
  865. disp = &fb_display[con];
  866. else
  867. disp = info->disp;
  868. if ((var->activate & FB_ACTIVATE_MASK) == FB_ACTIVATE_NOW) {
  869. if (disp->var.xres != var->xres ||
  870.     disp->var.yres != var->yres ||
  871.     disp->var.xres_virtual != var->xres_virtual ||
  872.     disp->var.yres_virtual != var->yres_virtual ||
  873.     disp->var.bits_per_pixel != var->bits_per_pixel ||
  874.     disp->var.accel_flags != var->accel_flags)
  875. return -EINVAL;
  876. }
  877. return 0;
  878. }
  879. static int
  880. stifb_get_cmap(struct fb_cmap *cmap, int kspc, int con,
  881.        struct fb_info *info)
  882. {
  883. struct stifb_info *fb = (struct stifb_info *)info;
  884. if (con == fb->currcon) /* current console ? */
  885. return fb_get_cmap(cmap, kspc, stifb_getcolreg, info);
  886. else if (fb_display[con].cmap.len) /* non default colormap ? */
  887. fb_copy_cmap(&fb_display[con].cmap, cmap, kspc ? 0 : 2);
  888. else
  889. fb_copy_cmap(fb_default_cmap(fb_display[con].var.bits_per_pixel > 8 ? 16 : 256), cmap, kspc ? 0: 2);
  890. return 0;
  891. }
  892. static int
  893. stifb_set_cmap(struct fb_cmap *cmap, int kspc, int con,
  894.        struct fb_info *info)
  895. {
  896. struct stifb_info *fb = (struct stifb_info *)info;
  897. struct display *disp;
  898. int err;
  899. if (con >= 0)
  900. disp = &fb_display[con];
  901. else
  902. disp = info->disp;
  903. if (!disp->cmap.len) { /* no colormap allocated ? */
  904. if ((err = fb_alloc_cmap(&disp->cmap, disp->var.bits_per_pixel > 8 ? 16 : 256, 0)))
  905. return err;
  906. }
  907. if (con == fb->currcon || con == -1) {
  908. err = fb_set_cmap(cmap, kspc, stifb_setcolreg, info);
  909. if (!err)
  910. stifb_loadcmap ((struct stifb_info *)info);
  911. return err;
  912. } else
  913. fb_copy_cmap(cmap, &disp->cmap, kspc ? 0 : 1);
  914. return 0;
  915. }
  916.  
  917. static void
  918. stifb_blank(int blank_mode, struct fb_info *info)
  919. {
  920. struct stifb_info *fb = (struct stifb_info *) info;
  921. int enable = (blank_mode == 0) ? ENABLE : DISABLE;
  922. switch (fb->id) {
  923. case S9000_ID_A1439A:
  924. CRX24_ENABLE_DISABLE_DISPLAY(fb, enable);
  925. break;
  926. case CRT_ID_VISUALIZE_EG:
  927. case S9000_ID_ARTIST:
  928. ARTIST_ENABLE_DISABLE_DISPLAY(fb, enable);
  929. break;
  930. case S9000_ID_HCRX:
  931. HYPER_ENABLE_DISABLE_DISPLAY(fb, enable);
  932. break;
  933. case S9000_ID_A1659A:; /* fall through */
  934. case S9000_ID_TIMBER:;
  935. case CRX24_OVERLAY_PLANES:;
  936. default:
  937. ENABLE_DISABLE_DISPLAY(fb, enable);
  938. break;
  939. }
  940. SETUP_FB(fb);
  941. }
  942. static void
  943. stifb_set_disp(struct stifb_info *fb)
  944. {
  945. int id = fb->id;
  946. SETUP_FB(fb);
  947. /* HCRX specific initialization */
  948. SETUP_HCRX(fb);
  949. /*
  950. if (id == S9000_ID_HCRX)
  951. hyperInitSprite(fb);
  952. else
  953. ngleInitSprite(fb);
  954. */
  955. /* Initialize the image planes. */ 
  956.         switch (id) {
  957.  case S9000_ID_HCRX:
  958.     hyperResetPlanes(fb, ENABLE);
  959.     break;
  960.  case S9000_ID_A1439A:
  961.     rattlerSetupPlanes(fb);
  962.     break;
  963.  case S9000_ID_A1659A:
  964.  case S9000_ID_ARTIST:
  965.  case CRT_ID_VISUALIZE_EG:
  966.     elkSetupPlanes(fb);
  967.     break;
  968. }
  969. /* Clear attribute planes on non HCRX devices. */
  970.         switch (id) {
  971.  case S9000_ID_A1659A:
  972.  case S9000_ID_A1439A:
  973.     if (fb->var.bits_per_pixel == 32)
  974. ngleSetupAttrPlanes(fb, BUFF1_CMAP3);
  975.     else {
  976. ngleSetupAttrPlanes(fb, BUFF1_CMAP0);
  977.     }
  978.     if (id == S9000_ID_A1439A)
  979. ngleClearOverlayPlanes(fb, 0xff, 0);
  980.     break;
  981.  case S9000_ID_ARTIST:
  982.  case CRT_ID_VISUALIZE_EG:
  983.     if (fb->var.bits_per_pixel == 32)
  984. ngleSetupAttrPlanes(fb, BUFF1_CMAP3);
  985.     else {
  986. ngleSetupAttrPlanes(fb, ARTIST_CMAP0);
  987.     }
  988.     break;
  989. }
  990. stifb_blank(0, (struct fb_info *)fb); /* 0=enable screen */
  991. SETUP_FB(fb);
  992. }
  993. static int
  994. stifb_switch(int con, struct fb_info *info)
  995. {
  996. struct stifb_info *fb = (struct stifb_info *)info;
  997. /* Do we have to save the colormap ? */
  998. if (fb->currcon != -1 && fb_display[fb->currcon].cmap.len)
  999. fb_get_cmap(&fb_display[fb->currcon].cmap, 1, stifb_getcolreg, info);
  1000. fb->currcon = con;
  1001. /* Install new colormap */
  1002. if (fb_display[con].cmap.len)
  1003. fb_set_cmap(&fb_display[con].cmap, 1, stifb_setcolreg, info);
  1004. else
  1005. fb_set_cmap(fb_default_cmap(fb_display[con].var.bits_per_pixel > 8 ? 16 : 256),
  1006.     1, stifb_setcolreg, info);
  1007. stifb_loadcmap ((struct stifb_info *)info);
  1008. return 0;
  1009. }
  1010. static int
  1011. stifb_update_var(int con, struct fb_info *info)
  1012. {
  1013. return 0;
  1014. }
  1015. /* ------------ Interfaces to hardware functions ------------ */
  1016. static struct fb_ops stifb_ops = {
  1017. owner: THIS_MODULE,
  1018. fb_get_fix: stifb_get_fix,
  1019. fb_get_var: stifb_get_var,
  1020. fb_set_var: stifb_set_var,
  1021. fb_get_cmap: stifb_get_cmap,
  1022. fb_set_cmap: stifb_set_cmap,
  1023. // fb_pan_display: fbgen_pan_display,
  1024. // fb_ioctl:       xxxfb_ioctl,   /* optional */      
  1025. };
  1026.     /*
  1027.      *  Initialization
  1028.      */
  1029. int __init
  1030. stifb_init_fb(struct sti_struct *sti, int force_bpp)
  1031. {
  1032. struct fb_fix_screeninfo *fix;
  1033. struct fb_var_screeninfo *var;
  1034. struct display *disp;
  1035. struct stifb_info *fb;
  1036. unsigned long sti_rom_address;
  1037. char *dev_name;
  1038. int bpp, xres, yres;
  1039. fb = kmalloc(sizeof(struct stifb_info), GFP_ATOMIC);
  1040. if (!fb) {
  1041. printk(KERN_ERR "stifb: Could not allocate stifb structuren");
  1042. return -ENODEV;
  1043. }
  1044. /* set struct to a known state */
  1045. memset(fb, 0, sizeof(struct stifb_info));
  1046. fix = &fb->fix;
  1047. var = &fb->var;
  1048. disp = &fb->disp;
  1049. fb->currcon = -1;
  1050. fb->cmap_reload = 1;
  1051. fb->sti = sti;
  1052. /* store upper 32bits of the graphics id */
  1053. fb->id = fb->sti->graphics_id[0];
  1054. fb->real_id = fb->id; /* save the real id */
  1055. /* only supported cards are allowed */
  1056. switch (fb->id) {
  1057. case S9000_ID_ARTIST:
  1058. case S9000_ID_HCRX:
  1059. case S9000_ID_TIMBER:
  1060. case S9000_ID_A1659A:
  1061. case S9000_ID_A1439A:
  1062. case CRT_ID_VISUALIZE_EG:
  1063. break;
  1064. default:
  1065. printk(KERN_WARNING "stifb: Unsupported gfx card id 0x%08xn",
  1066. fb->id);
  1067. goto out_err1;
  1068. }
  1069. /* default to 8 bpp on most graphic chips */
  1070. bpp = 8;
  1071. xres = sti_onscreen_x(fb->sti);
  1072. yres = sti_onscreen_y(fb->sti);
  1073. ngleGetDeviceRomData(fb);
  1074. /* get (virtual) io region base addr */
  1075. fix->mmio_start = REGION_BASE(fb,2);
  1076. fix->mmio_len   = 0x400000;
  1077.         /* Reject any device not in the NGLE family */
  1078. switch (fb->id) {
  1079. case S9000_ID_A1659A: /* CRX/A1659A */
  1080. break;
  1081. case S9000_ID_ELM: /* GRX, grayscale but else same as A1659A */
  1082. var->grayscale = 1;
  1083. fb->id = S9000_ID_A1659A;
  1084. break;
  1085. case S9000_ID_TIMBER: /* HP9000/710 Any (may be a grayscale device) */
  1086. dev_name = fb->sti->outptr.dev_name;
  1087. if (strstr(dev_name, "GRAYSCALE") || 
  1088.     strstr(dev_name, "Grayscale") ||
  1089.     strstr(dev_name, "grayscale"))
  1090. var->grayscale = 1;
  1091. break;
  1092. case S9000_ID_TOMCAT: /* Dual CRX, behaves else like a CRX */
  1093. /* FIXME: TomCat supports two heads:
  1094.  * fb.iobase = REGION_BASE(fb_info,3);
  1095.  * fb.screen_base = (void*) REGION_BASE(fb_info,2);
  1096.  * for now we only support the left one ! */
  1097. xres = fb->ngle_rom.x_size_visible;
  1098. yres = fb->ngle_rom.y_size_visible;
  1099. fb->id = S9000_ID_A1659A;
  1100. break;
  1101. case S9000_ID_A1439A: /* CRX24/A1439A */
  1102. bpp = 32;
  1103. break;
  1104. case S9000_ID_HCRX: /* Hyperdrive/HCRX */
  1105. memset(&fb->ngle_rom, 0, sizeof(fb->ngle_rom));
  1106. if ((fb->sti->regions_phys[0] & 0xfc000000) ==
  1107.     (fb->sti->regions_phys[2] & 0xfc000000))
  1108. sti_rom_address = fb->sti->regions_phys[0];
  1109. else
  1110. sti_rom_address = fb->sti->regions_phys[1];
  1111. #ifdef __LP64__
  1112.         sti_rom_address |= 0xffffffff00000000;
  1113. #endif
  1114. fb->deviceSpecificConfig = __raw_readl(sti_rom_address);
  1115. if (IS_24_DEVICE(fb)) {
  1116. if (force_bpp == 8 || force_bpp == 32)
  1117. bpp = force_bpp;
  1118. else
  1119. bpp = 32;
  1120. } else
  1121. bpp = 8;
  1122. READ_WORD(fb, REG_15);
  1123. SETUP_HW(fb);
  1124. break;
  1125. case CRT_ID_VISUALIZE_EG:
  1126. case S9000_ID_ARTIST: /* Artist */
  1127. break;
  1128. default: 
  1129. #ifdef FALLBACK_TO_1BPP
  1130.         printk(KERN_WARNING 
  1131. "stifb: Unsupported graphics card (id=0x%08x) "
  1132. "- now trying 1bpp mode insteadn",
  1133. fb->id);
  1134. bpp = 1; /* default to 1 bpp */
  1135. break;
  1136. #else
  1137.         printk(KERN_WARNING 
  1138. "stifb: Unsupported graphics card (id=0x%08x) "
  1139. "- skipping.n",
  1140. fb->id);
  1141. goto out_err1;
  1142. #endif
  1143. }
  1144. /* get framebuffer pysical and virtual base addr & len (64bit ready) */
  1145. fix->smem_start = fb->sti->regions_phys[1] | 0xffffffff00000000;
  1146. fix->smem_len = fb->sti->regions[1].region_desc.length * 4096;
  1147. fix->line_length = (fb->sti->glob_cfg->total_x * bpp) / 8;
  1148. if (!fix->line_length)
  1149. fix->line_length = 2048; /* default */
  1150. fix->accel = FB_ACCEL_NONE;
  1151. switch (bpp) {
  1152.     case 1:
  1153. fix->type = FB_TYPE_PLANES; /* well, sort of */
  1154. fix->visual = FB_VISUAL_MONO10;
  1155. disp->dispsw = &fbcon_sti;
  1156. break;
  1157. #ifdef FBCON_HAS_CFB8
  1158.     case 8:
  1159. fix->type = FB_TYPE_PACKED_PIXELS;
  1160. fix->visual = FB_VISUAL_PSEUDOCOLOR;
  1161.   disp->dispsw = &fbcon_cfb8;
  1162. var->red.length = var->green.length = var->blue.length = 8;
  1163. break;
  1164. #endif
  1165. #ifdef FBCON_HAS_CFB32
  1166.     case 32:
  1167. fix->type = FB_TYPE_PACKED_PIXELS;
  1168. fix->visual = FB_VISUAL_TRUECOLOR;
  1169. disp->dispsw = &fbcon_cfb32;
  1170. disp->dispsw_data = fb->fbcon_cmap.cfb32;
  1171. var->red.length = var->green.length = var->blue.length = var->transp.length = 8;
  1172. var->blue.offset = 0;
  1173. var->green.offset = 8;
  1174. var->red.offset = 16;
  1175. var->transp.offset = 24;
  1176. break;
  1177. #endif
  1178.     default:
  1179. disp->dispsw = &fbcon_dummy;
  1180. break;
  1181. }
  1182. var->xres = var->xres_virtual = xres;
  1183. var->yres = var->yres_virtual = yres;
  1184. var->bits_per_pixel = bpp;
  1185. disp->var = *var;
  1186. disp->visual = fix->visual;
  1187. disp->type = fix->type;
  1188. disp->type_aux = fix->type_aux;
  1189. disp->line_length = fix->line_length;
  1190. disp->var.activate = FB_ACTIVATE_NOW;
  1191. disp->screen_base = (void*) REGION_BASE(fb,1);
  1192. disp->can_soft_blank = 1;
  1193. disp->scrollmode = SCROLL_YREDRAW;
  1194. strcpy(fb->info.modename, "stifb");
  1195. fb->info.node = -1;
  1196. fb->info.flags = FBINFO_FLAG_DEFAULT;
  1197. fb->info.fbops = &stifb_ops;
  1198. fb->info.disp = disp;
  1199. fb->info.changevar = NULL;
  1200. fb->info.switch_con = &stifb_switch;
  1201. fb->info.updatevar = &stifb_update_var;
  1202. fb->info.blank = &stifb_blank;
  1203. fb->info.flags = FBINFO_FLAG_DEFAULT;
  1204. stifb_set_var(&disp->var, 1, &fb->info);
  1205. stifb_set_disp(fb);
  1206. if (!request_mem_region(fix->smem_start, fix->smem_len, "stifb")) {
  1207. printk(KERN_ERR "stifb: cannot reserve fb region 0x%04lx-0x%04lxn",
  1208. fix->smem_start, fix->smem_start+fix->smem_len);
  1209. goto out_err1;
  1210. }
  1211. if (!request_mem_region(fix->mmio_start, fix->mmio_len, "stifb mmio")) {
  1212. printk(KERN_ERR "stifb: cannot reserve sti mmio region 0x%04lx-0x%04lxn",
  1213. fix->mmio_start, fix->mmio_start+fix->mmio_len);
  1214. goto out_err2;
  1215. }
  1216. if (register_framebuffer(&fb->info) < 0)
  1217. goto out_err3;
  1218. printk(KERN_INFO 
  1219.     "fb%d: %s %dx%d-%d frame buffer device, id: %04x, mmio: 0x%04lxn",
  1220. GET_FB_IDX(fb->info.node), 
  1221. fb->info.modename,
  1222. disp->var.xres, 
  1223. disp->var.yres,
  1224. disp->var.bits_per_pixel,
  1225. fb->id, 
  1226. fix->mmio_start);
  1227. return 0;
  1228. out_err3:
  1229. release_mem_region(fix->mmio_start, fix->mmio_len);
  1230. out_err2:
  1231. release_mem_region(fix->smem_start, fix->smem_len);
  1232. out_err1:
  1233. kfree(fb);
  1234. return -ENXIO;
  1235. }
  1236. int __init
  1237. stifb_init(void)
  1238. {
  1239. struct sti_struct *sti;
  1240. int i;
  1241. if (sti_init_roms() == NULL)
  1242. return -ENXIO; /* no STI cards available */
  1243. for (i = 0; i < MAX_STI_ROMS; i++) {
  1244. sti = sti_get_rom(i);
  1245. if (sti)
  1246. stifb_init_fb (sti, stifb_force_bpp[i]);
  1247. else
  1248. break;
  1249. }
  1250. return 0;
  1251. }
  1252. /*
  1253.  *  Cleanup
  1254.  */
  1255. void __exit
  1256. stifb_cleanup(struct fb_info *info)
  1257. {
  1258. // unregister_framebuffer(info); 
  1259. }
  1260. int __init
  1261. stifb_setup(char *options)
  1262. {
  1263. int i;
  1264. if (!options || !*options)
  1265. return 0;
  1266. if (strncmp(options, "bpp", 3) == 0) {
  1267. options += 3;
  1268. for (i = 0; i < MAX_STI_ROMS; i++) {
  1269. if (*options++ == ':')
  1270. stifb_force_bpp[i] = simple_strtoul(options, &options, 10);
  1271. else
  1272. break;
  1273. }
  1274. }
  1275. return 0;
  1276. }
  1277. __setup("stifb=", stifb_setup);
  1278. #ifdef MODULE
  1279. module_init(stifb_init);
  1280. #endif
  1281. module_exit(stifb_cleanup);
  1282. MODULE_AUTHOR("Helge Deller <deller@gmx.de>, Thomas Bogendoerfer <tsbogend@alpha.franken.de>");
  1283. MODULE_DESCRIPTION("Framebuffer driver for HP's NGLE series graphics cards in HP PARISC machines");
  1284. MODULE_LICENSE("GPL");
  1285. MODULE_PARM(bpp, "i");
  1286. MODULE_PARM_DESC(mem, "Bits per pixel (default: 8)");