stifb.c 35 KB

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