stifb.c 35 KB

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