stifb.c 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485
  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-2005 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(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. #define TMPBUFLEN 2048
  756. static ssize_t
  757. stifb_read(struct file *file, char *buf, size_t count, loff_t *ppos)
  758. {
  759. unsigned long p = *ppos;
  760. struct inode *inode = file->f_dentry->d_inode;
  761. int fbidx = iminor(inode);
  762. struct fb_info *info = registered_fb[fbidx];
  763. char tmpbuf[TMPBUFLEN];
  764. if (!info || ! info->screen_base)
  765. return -ENODEV;
  766. if (p >= info->fix.smem_len)
  767. return 0;
  768. if (count >= info->fix.smem_len)
  769. count = info->fix.smem_len;
  770. if (count + p > info->fix.smem_len)
  771. count = info->fix.smem_len - p;
  772. if (count > sizeof(tmpbuf))
  773. count = sizeof(tmpbuf);
  774. if (count) {
  775. char *base_addr;
  776. base_addr = info->screen_base;
  777. memcpy_fromio(&tmpbuf, base_addr+p, count);
  778. count -= copy_to_user(buf, &tmpbuf, count);
  779. if (!count)
  780. return -EFAULT;
  781. *ppos += count;
  782. }
  783. return count;
  784. }
  785. static ssize_t
  786. stifb_write(struct file *file, const char *buf, size_t count, loff_t *ppos)
  787. {
  788. struct inode *inode = file->f_dentry->d_inode;
  789. int fbidx = iminor(inode);
  790. struct fb_info *info = registered_fb[fbidx];
  791. unsigned long p = *ppos;
  792. size_t c;
  793. int err;
  794. char tmpbuf[TMPBUFLEN];
  795. if (!info || !info->screen_base)
  796. return -ENODEV;
  797. if (p > info->fix.smem_len)
  798. return -ENOSPC;
  799. if (count >= info->fix.smem_len)
  800. count = info->fix.smem_len;
  801. err = 0;
  802. if (count + p > info->fix.smem_len) {
  803. count = info->fix.smem_len - p;
  804. err = -ENOSPC;
  805. }
  806. p += (unsigned long)info->screen_base;
  807. c = count;
  808. while (c) {
  809. int len = c > sizeof(tmpbuf) ? sizeof(tmpbuf) : c;
  810. err = -EFAULT;
  811. if (copy_from_user(&tmpbuf, buf, len))
  812. break;
  813. memcpy_toio(p, &tmpbuf, len);
  814. c -= len;
  815. p += len;
  816. buf += len;
  817. *ppos += len;
  818. }
  819. if (count-c)
  820. return (count-c);
  821. return err;
  822. }
  823. static int
  824. stifb_setcolreg(u_int regno, u_int red, u_int green,
  825. u_int blue, u_int transp, struct fb_info *info)
  826. {
  827. struct stifb_info *fb = (struct stifb_info *) info;
  828. u32 color;
  829. if (regno >= NR_PALETTE)
  830. return 1;
  831. red >>= 8;
  832. green >>= 8;
  833. blue >>= 8;
  834. DEBUG_OFF();
  835. START_IMAGE_COLORMAP_ACCESS(fb);
  836. if (unlikely(fb->info.var.grayscale)) {
  837. /* gray = 0.30*R + 0.59*G + 0.11*B */
  838. color = ((red * 77) +
  839. (green * 151) +
  840. (blue * 28)) >> 8;
  841. } else {
  842. color = ((red << 16) |
  843. (green << 8) |
  844. (blue));
  845. }
  846. if (fb->info.fix.visual == FB_VISUAL_DIRECTCOLOR) {
  847. struct fb_var_screeninfo *var = &fb->info.var;
  848. if (regno < 16)
  849. ((u32 *)fb->info.pseudo_palette)[regno] =
  850. regno << var->red.offset |
  851. regno << var->green.offset |
  852. regno << var->blue.offset;
  853. }
  854. WRITE_IMAGE_COLOR(fb, regno, color);
  855. if (fb->id == S9000_ID_HCRX) {
  856. NgleLutBltCtl lutBltCtl;
  857. lutBltCtl = setHyperLutBltCtl(fb,
  858. 0, /* Offset w/i LUT */
  859. 256); /* Load entire LUT */
  860. NGLE_BINC_SET_SRCADDR(fb,
  861. NGLE_LONG_FB_ADDRESS(0, 0x100, 0));
  862. /* 0x100 is same as used in WRITE_IMAGE_COLOR() */
  863. START_COLORMAPLOAD(fb, lutBltCtl.all);
  864. SETUP_FB(fb);
  865. } else {
  866. /* cleanup colormap hardware */
  867. FINISH_IMAGE_COLORMAP_ACCESS(fb);
  868. }
  869. DEBUG_ON();
  870. return 0;
  871. }
  872. static int
  873. stifb_blank(int blank_mode, struct fb_info *info)
  874. {
  875. struct stifb_info *fb = (struct stifb_info *) info;
  876. int enable = (blank_mode == 0) ? ENABLE : DISABLE;
  877. switch (fb->id) {
  878. case S9000_ID_A1439A:
  879. CRX24_ENABLE_DISABLE_DISPLAY(fb, enable);
  880. break;
  881. case CRT_ID_VISUALIZE_EG:
  882. case S9000_ID_ARTIST:
  883. ARTIST_ENABLE_DISABLE_DISPLAY(fb, enable);
  884. break;
  885. case S9000_ID_HCRX:
  886. HYPER_ENABLE_DISABLE_DISPLAY(fb, enable);
  887. break;
  888. case S9000_ID_A1659A: /* fall through */
  889. case S9000_ID_TIMBER:
  890. case CRX24_OVERLAY_PLANES:
  891. default:
  892. ENABLE_DISABLE_DISPLAY(fb, enable);
  893. break;
  894. }
  895. SETUP_FB(fb);
  896. return 0;
  897. }
  898. static void __init
  899. stifb_init_display(struct stifb_info *fb)
  900. {
  901. int id = fb->id;
  902. SETUP_FB(fb);
  903. /* HCRX specific initialization */
  904. SETUP_HCRX(fb);
  905. /*
  906. if (id == S9000_ID_HCRX)
  907. hyperInitSprite(fb);
  908. else
  909. ngleInitSprite(fb);
  910. */
  911. /* Initialize the image planes. */
  912. switch (id) {
  913. case S9000_ID_HCRX:
  914. hyperResetPlanes(fb, ENABLE);
  915. break;
  916. case S9000_ID_A1439A:
  917. rattlerSetupPlanes(fb);
  918. break;
  919. case S9000_ID_A1659A:
  920. case S9000_ID_ARTIST:
  921. case CRT_ID_VISUALIZE_EG:
  922. elkSetupPlanes(fb);
  923. break;
  924. }
  925. /* Clear attribute planes on non HCRX devices. */
  926. switch (id) {
  927. case S9000_ID_A1659A:
  928. case S9000_ID_A1439A:
  929. if (fb->info.var.bits_per_pixel == 32)
  930. ngleSetupAttrPlanes(fb, BUFF1_CMAP3);
  931. else {
  932. ngleSetupAttrPlanes(fb, BUFF1_CMAP0);
  933. }
  934. if (id == S9000_ID_A1439A)
  935. ngleClearOverlayPlanes(fb, 0xff, 0);
  936. break;
  937. case S9000_ID_ARTIST:
  938. case CRT_ID_VISUALIZE_EG:
  939. if (fb->info.var.bits_per_pixel == 32)
  940. ngleSetupAttrPlanes(fb, BUFF1_CMAP3);
  941. else {
  942. ngleSetupAttrPlanes(fb, ARTIST_CMAP0);
  943. }
  944. break;
  945. }
  946. stifb_blank(0, (struct fb_info *)fb); /* 0=enable screen */
  947. SETUP_FB(fb);
  948. }
  949. /* ------------ Interfaces to hardware functions ------------ */
  950. static struct fb_ops stifb_ops = {
  951. .owner = THIS_MODULE,
  952. .fb_read = stifb_read,
  953. .fb_write = stifb_write,
  954. .fb_setcolreg = stifb_setcolreg,
  955. .fb_blank = stifb_blank,
  956. .fb_fillrect = cfb_fillrect,
  957. .fb_copyarea = cfb_copyarea,
  958. .fb_imageblit = cfb_imageblit,
  959. };
  960. /*
  961. * Initialization
  962. */
  963. int __init
  964. stifb_init_fb(struct sti_struct *sti, int bpp_pref)
  965. {
  966. struct fb_fix_screeninfo *fix;
  967. struct fb_var_screeninfo *var;
  968. struct stifb_info *fb;
  969. struct fb_info *info;
  970. unsigned long sti_rom_address;
  971. char *dev_name;
  972. int bpp, xres, yres;
  973. fb = kmalloc(sizeof(*fb), GFP_ATOMIC);
  974. if (!fb) {
  975. printk(KERN_ERR "stifb: Could not allocate stifb structure\n");
  976. return -ENODEV;
  977. }
  978. info = &fb->info;
  979. /* set struct to a known state */
  980. memset(fb, 0, sizeof(*fb));
  981. fix = &info->fix;
  982. var = &info->var;
  983. fb->sti = sti;
  984. /* store upper 32bits of the graphics id */
  985. fb->id = fb->sti->graphics_id[0];
  986. /* only supported cards are allowed */
  987. switch (fb->id) {
  988. case CRT_ID_VISUALIZE_EG:
  989. /* look for a double buffering device like e.g. the
  990. "INTERNAL_EG_DX1024" in the RDI precisionbook laptop
  991. which won't work. The same device in non-double
  992. buffering mode returns "INTERNAL_EG_X1024". */
  993. if (strstr(sti->outptr.dev_name, "EG_DX")) {
  994. printk(KERN_WARNING
  995. "stifb: ignoring '%s'. Disable double buffering in IPL menu.\n",
  996. sti->outptr.dev_name);
  997. goto out_err0;
  998. }
  999. /* fall though */
  1000. case S9000_ID_ARTIST:
  1001. case S9000_ID_HCRX:
  1002. case S9000_ID_TIMBER:
  1003. case S9000_ID_A1659A:
  1004. case S9000_ID_A1439A:
  1005. break;
  1006. default:
  1007. printk(KERN_WARNING "stifb: '%s' (id: 0x%08x) not supported.\n",
  1008. sti->outptr.dev_name, fb->id);
  1009. goto out_err0;
  1010. }
  1011. /* default to 8 bpp on most graphic chips */
  1012. bpp = 8;
  1013. xres = sti_onscreen_x(fb->sti);
  1014. yres = sti_onscreen_y(fb->sti);
  1015. ngleGetDeviceRomData(fb);
  1016. /* get (virtual) io region base addr */
  1017. fix->mmio_start = REGION_BASE(fb,2);
  1018. fix->mmio_len = 0x400000;
  1019. /* Reject any device not in the NGLE family */
  1020. switch (fb->id) {
  1021. case S9000_ID_A1659A: /* CRX/A1659A */
  1022. break;
  1023. case S9000_ID_ELM: /* GRX, grayscale but else same as A1659A */
  1024. var->grayscale = 1;
  1025. fb->id = S9000_ID_A1659A;
  1026. break;
  1027. case S9000_ID_TIMBER: /* HP9000/710 Any (may be a grayscale device) */
  1028. dev_name = fb->sti->outptr.dev_name;
  1029. if (strstr(dev_name, "GRAYSCALE") ||
  1030. strstr(dev_name, "Grayscale") ||
  1031. strstr(dev_name, "grayscale"))
  1032. var->grayscale = 1;
  1033. break;
  1034. case S9000_ID_TOMCAT: /* Dual CRX, behaves else like a CRX */
  1035. /* FIXME: TomCat supports two heads:
  1036. * fb.iobase = REGION_BASE(fb_info,3);
  1037. * fb.screen_base = (void*) REGION_BASE(fb_info,2);
  1038. * for now we only support the left one ! */
  1039. xres = fb->ngle_rom.x_size_visible;
  1040. yres = fb->ngle_rom.y_size_visible;
  1041. fb->id = S9000_ID_A1659A;
  1042. break;
  1043. case S9000_ID_A1439A: /* CRX24/A1439A */
  1044. bpp = 32;
  1045. break;
  1046. case S9000_ID_HCRX: /* Hyperdrive/HCRX */
  1047. memset(&fb->ngle_rom, 0, sizeof(fb->ngle_rom));
  1048. if ((fb->sti->regions_phys[0] & 0xfc000000) ==
  1049. (fb->sti->regions_phys[2] & 0xfc000000))
  1050. sti_rom_address = F_EXTEND(fb->sti->regions_phys[0]);
  1051. else
  1052. sti_rom_address = F_EXTEND(fb->sti->regions_phys[1]);
  1053. fb->deviceSpecificConfig = gsc_readl(sti_rom_address);
  1054. if (IS_24_DEVICE(fb)) {
  1055. if (bpp_pref == 8 || bpp_pref == 32)
  1056. bpp = bpp_pref;
  1057. else
  1058. bpp = 32;
  1059. } else
  1060. bpp = 8;
  1061. READ_WORD(fb, REG_15);
  1062. SETUP_HW(fb);
  1063. break;
  1064. case CRT_ID_VISUALIZE_EG:
  1065. case S9000_ID_ARTIST: /* Artist */
  1066. break;
  1067. default:
  1068. #ifdef FALLBACK_TO_1BPP
  1069. printk(KERN_WARNING
  1070. "stifb: Unsupported graphics card (id=0x%08x) "
  1071. "- now trying 1bpp mode instead\n",
  1072. fb->id);
  1073. bpp = 1; /* default to 1 bpp */
  1074. break;
  1075. #else
  1076. printk(KERN_WARNING
  1077. "stifb: Unsupported graphics card (id=0x%08x) "
  1078. "- skipping.\n",
  1079. fb->id);
  1080. goto out_err0;
  1081. #endif
  1082. }
  1083. /* get framebuffer physical and virtual base addr & len (64bit ready) */
  1084. fix->smem_start = F_EXTEND(fb->sti->regions_phys[1]);
  1085. fix->smem_len = fb->sti->regions[1].region_desc.length * 4096;
  1086. fix->line_length = (fb->sti->glob_cfg->total_x * bpp) / 8;
  1087. if (!fix->line_length)
  1088. fix->line_length = 2048; /* default */
  1089. /* limit fbsize to max visible screen size */
  1090. if (fix->smem_len > yres*fix->line_length)
  1091. fix->smem_len = yres*fix->line_length;
  1092. fix->accel = FB_ACCEL_NONE;
  1093. switch (bpp) {
  1094. case 1:
  1095. fix->type = FB_TYPE_PLANES; /* well, sort of */
  1096. fix->visual = FB_VISUAL_MONO10;
  1097. var->red.length = var->green.length = var->blue.length = 1;
  1098. break;
  1099. case 8:
  1100. fix->type = FB_TYPE_PACKED_PIXELS;
  1101. fix->visual = FB_VISUAL_PSEUDOCOLOR;
  1102. var->red.length = var->green.length = var->blue.length = 8;
  1103. break;
  1104. case 32:
  1105. fix->type = FB_TYPE_PACKED_PIXELS;
  1106. fix->visual = FB_VISUAL_DIRECTCOLOR;
  1107. var->red.length = var->green.length = var->blue.length = var->transp.length = 8;
  1108. var->blue.offset = 0;
  1109. var->green.offset = 8;
  1110. var->red.offset = 16;
  1111. var->transp.offset = 24;
  1112. break;
  1113. default:
  1114. break;
  1115. }
  1116. var->xres = var->xres_virtual = xres;
  1117. var->yres = var->yres_virtual = yres;
  1118. var->bits_per_pixel = bpp;
  1119. strcpy(fix->id, "stifb");
  1120. info->fbops = &stifb_ops;
  1121. info->screen_base = (void*) REGION_BASE(fb,1);
  1122. info->flags = FBINFO_DEFAULT;
  1123. info->pseudo_palette = &fb->pseudo_palette;
  1124. /* This has to been done !!! */
  1125. fb_alloc_cmap(&info->cmap, NR_PALETTE, 0);
  1126. stifb_init_display(fb);
  1127. if (!request_mem_region(fix->smem_start, fix->smem_len, "stifb fb")) {
  1128. printk(KERN_ERR "stifb: cannot reserve fb region 0x%04lx-0x%04lx\n",
  1129. fix->smem_start, fix->smem_start+fix->smem_len);
  1130. goto out_err1;
  1131. }
  1132. if (!request_mem_region(fix->mmio_start, fix->mmio_len, "stifb mmio")) {
  1133. printk(KERN_ERR "stifb: cannot reserve sti mmio region 0x%04lx-0x%04lx\n",
  1134. fix->mmio_start, fix->mmio_start+fix->mmio_len);
  1135. goto out_err2;
  1136. }
  1137. if (register_framebuffer(&fb->info) < 0)
  1138. goto out_err3;
  1139. sti->info = info; /* save for unregister_framebuffer() */
  1140. printk(KERN_INFO
  1141. "fb%d: %s %dx%d-%d frame buffer device, %s, id: %04x, mmio: 0x%04lx\n",
  1142. fb->info.node,
  1143. fix->id,
  1144. var->xres,
  1145. var->yres,
  1146. var->bits_per_pixel,
  1147. sti->outptr.dev_name,
  1148. fb->id,
  1149. fix->mmio_start);
  1150. return 0;
  1151. out_err3:
  1152. release_mem_region(fix->mmio_start, fix->mmio_len);
  1153. out_err2:
  1154. release_mem_region(fix->smem_start, fix->smem_len);
  1155. out_err1:
  1156. fb_dealloc_cmap(&info->cmap);
  1157. out_err0:
  1158. kfree(fb);
  1159. return -ENXIO;
  1160. }
  1161. static int stifb_disabled __initdata;
  1162. int __init
  1163. stifb_setup(char *options);
  1164. int __init
  1165. stifb_init(void)
  1166. {
  1167. struct sti_struct *sti;
  1168. struct sti_struct *def_sti;
  1169. int i;
  1170. #ifndef MODULE
  1171. char *option = NULL;
  1172. if (fb_get_options("stifb", &option))
  1173. return -ENODEV;
  1174. stifb_setup(option);
  1175. #endif
  1176. if (stifb_disabled) {
  1177. printk(KERN_INFO "stifb: disabled by \"stifb=off\" kernel parameter\n");
  1178. return -ENXIO;
  1179. }
  1180. def_sti = sti_get_rom(0);
  1181. if (def_sti) {
  1182. for (i = 1; i <= MAX_STI_ROMS; i++) {
  1183. sti = sti_get_rom(i);
  1184. if (!sti)
  1185. break;
  1186. if (sti == def_sti) {
  1187. stifb_init_fb(sti, stifb_bpp_pref[i - 1]);
  1188. break;
  1189. }
  1190. }
  1191. }
  1192. for (i = 1; i <= MAX_STI_ROMS; i++) {
  1193. sti = sti_get_rom(i);
  1194. if (!sti)
  1195. break;
  1196. if (sti == def_sti)
  1197. continue;
  1198. stifb_init_fb(sti, stifb_bpp_pref[i - 1]);
  1199. }
  1200. return 0;
  1201. }
  1202. /*
  1203. * Cleanup
  1204. */
  1205. static void __exit
  1206. stifb_cleanup(void)
  1207. {
  1208. struct sti_struct *sti;
  1209. int i;
  1210. for (i = 1; i <= MAX_STI_ROMS; i++) {
  1211. sti = sti_get_rom(i);
  1212. if (!sti)
  1213. break;
  1214. if (sti->info) {
  1215. struct fb_info *info = sti->info;
  1216. unregister_framebuffer(sti->info);
  1217. release_mem_region(info->fix.mmio_start, info->fix.mmio_len);
  1218. release_mem_region(info->fix.smem_start, info->fix.smem_len);
  1219. fb_dealloc_cmap(&info->cmap);
  1220. kfree(info);
  1221. }
  1222. sti->info = NULL;
  1223. }
  1224. }
  1225. int __init
  1226. stifb_setup(char *options)
  1227. {
  1228. int i;
  1229. if (!options || !*options)
  1230. return 0;
  1231. if (strncmp(options, "off", 3) == 0) {
  1232. stifb_disabled = 1;
  1233. options += 3;
  1234. }
  1235. if (strncmp(options, "bpp", 3) == 0) {
  1236. options += 3;
  1237. for (i = 0; i < MAX_STI_ROMS; i++) {
  1238. if (*options++ != ':')
  1239. break;
  1240. stifb_bpp_pref[i] = simple_strtoul(options, &options, 10);
  1241. }
  1242. }
  1243. return 0;
  1244. }
  1245. __setup("stifb=", stifb_setup);
  1246. module_init(stifb_init);
  1247. module_exit(stifb_cleanup);
  1248. MODULE_AUTHOR("Helge Deller <deller@gmx.de>, Thomas Bogendoerfer <tsbogend@alpha.franken.de>");
  1249. MODULE_DESCRIPTION("Framebuffer driver for HP's NGLE series graphics cards in HP PARISC machines");
  1250. MODULE_LICENSE("GPL v2");