I386.Raster.Mod 112 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615
  1. MODULE Raster; (** non-portable *) (* eos, TF **)
  2. (** AUTHOR "eos"; PURPOSE "Raster operations"; *)
  3. (**
  4. Raster image bitmaps and basic image processing
  5. **)
  6. (*
  7. 19.9.1999 - spawned from GfxMaps
  8. 25.10.1999 - fixed bytes-per-row calculation in Create (need to make arguments LONG)
  9. 17.11.1999 - eliminated F8 format, replaced by D8 (implemented with module Colors)
  10. 19.11.1999 - fixed missing alpha component in computed palettes
  11. 16.05.2000 - module Raster as Oberon-independent part of Images
  12. 19.06.2000 - replaced DisplayFormat and InitDisplay by DisplayFormat()
  13. 25.02.2006 - raster operations with SSE2 added (student project by Myrto Zehnder)
  14. 28.02.2008 - added capabilities for 16-bit palette & grayscale images often used in scientific,medical imaging and professional photography (Patrick Hunziker)
  15. To do:
  16. - store in compressed format
  17. - add capabilities for RGB and multiplane images with >8 bits per color as used in scientific imaging and professional photography
  18. *)
  19. IMPORT
  20. SYSTEM, KernelLog, Machine, Streams, CLUTs, Displays;
  21. (*
  22. Interfacing with display hardware and foreign framebuffers may suffer a performance hit if their
  23. bit or byte ordering can't be made compatible with that of a built-in format and has to be converted
  24. manually. Predefined format A1 has the leftmost pixel in the least significant bit of a byte, and
  25. all hi/true color formats have their blue component at the lowest address, followed by green,
  26. red, and possibly alpha (which conforms to the specification of the transfer formats in Displays).
  27. As SYSTEM is imported, the module is not portable and has always to be compiled to native code.
  28. However, it should usually suffice to recompile the module on other platforms without changing
  29. any code.
  30. Assumptions:
  31. * CHR(l) = CHR(l MOD 100H) for all l: LONGINT
  32. * SIZEOF(LONGINT)=4
  33. *)
  34. CONST
  35. b* = 0; g* = 1; r* = 2; a* = 3; (** index of blue, green, red, and alpha components in a Pixel **)
  36. (** format codes **)
  37. custom* = 0; a1* = 1; a8* = 2; d8* = 3; p8* = 4; bgr555* = 5; bgr565* = 6; bgr466* = 7; bgr888* = 8; bgra8888* = 9; p16* =10;
  38. (** components **)
  39. color* = 0; alpha* = 1; index* = 2;
  40. (** compositing operations (srcCopy = replace, srcOverDst = paint **)
  41. clear* = 0; srcCopy* = 1; dstCopy* = 2; srcOverDst* = 3; dstOverSrc* = 4; srcInDst* = 5; dstInSrc* = 6;
  42. srcWithoutDst* = 7; dstWithoutSrc* = 8; srcAtopDst* = 9; dstAtopSrc* = 10; srcXorDst* = 11; InvDst*=12;
  43. InvOverDst*=13;
  44. MAXCOL = 10000H; (*current implementation limitation for number of color indexes *)
  45. TYPE
  46. (** general pixels with red, green, blue, and alpha information in range 0..255; alpha is pre-multiplied into RGB **)
  47. Pixel* = ARRAY 4 OF CHAR;
  48. (** palette structure for indexed formats **)
  49. Palette* = OBJECT
  50. VAR
  51. col*: POINTER TO ARRAY OF Pixel; (** color table **)
  52. used*: LONGINT; (** number of valid entries in color table **)
  53. clut: CLUTs.CLUT; (* reverse color lookup structure *)
  54. PROCEDURE &New*; BEGIN NEW(col,256); used:=256 END New; (*initialized to 256 colors; for backwards compatibility*)
  55. PROCEDURE Init*(used:LONGINT); BEGIN SELF.used:=used; NEW(col,used) END Init; (*initialize to size # 256*) (*bugfix PH090122*)
  56. END Palette;
  57. (** image format **)
  58. Format0* = RECORD
  59. code*: SHORTINT; (** format code for quick format checks **)
  60. bpp*: SHORTINT; (** number of bits per pixel **)
  61. align*: SHORTINT; (** bytes per row must be multiple of this **)
  62. components*: SET; (** components that are stored in a pixel **)
  63. pal*: Palette; (** optional palette for indexed formats **)
  64. END;
  65. PackProc* = PROCEDURE (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  66. Format* = RECORD (Format0)
  67. pack*: PackProc; (** store supported pixel components at given address **)
  68. unpack*: PackProc; (** load supported pixel components from given address **)
  69. END;
  70. (** raster image **)
  71. Image* = OBJECT (* POINTER TO ImageDesc;
  72. ImageDesc* = RECORD *)
  73. VAR
  74. width*, height*: LONGINT; (** image dimensions **)
  75. fmt*: Format; (** pixel format **)
  76. bpr*: LONGINT; (** number of bytes per row (may be negative) **)
  77. adr*: ADDRESS; (** address of lower left pixel **)
  78. mem*: POINTER TO ARRAY OF CHAR; (** block where pixels are stored; mem#NIL implies adr=ADR(mem[0]) **)
  79. END Image;
  80. (** transfer mode **)
  81. Mode0* = RECORD
  82. src*, dst*: Format; (** source and destination format **)
  83. op*: LONGINT; (** compositing operation **)
  84. col*: Pixel; (** substitute color used when transfering from pure alpha formats to colored ones **)
  85. buf: Pixel; (* constant area for special-case moving *)
  86. map: POINTER TO ARRAY OF INTEGER; (* color map for transfer between indexed formats *)
  87. END;
  88. TransferProc* = PROCEDURE (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  89. Mode* = RECORD (Mode0)
  90. transfer*: TransferProc; (** procedure transfering pixels from source to destination **)
  91. END;
  92. PictureTransferParameters* = POINTER TO RECORD
  93. img* : Image;
  94. name* : ARRAY 128 OF CHAR;
  95. done* : BOOLEAN
  96. END;
  97. VAR
  98. A1*, A8*, D8*, BGR555*, BGR565*, BGR466*, BGR888*, BGRA8888*: Format; (** predefined formats **)
  99. PixelFormat*: Format; (** special formats **)
  100. Clamp*: ARRAY 500H OF CHAR; (** Clamp[200H+i] = CHR(min(max(i, 0), 0FFH)) **)
  101. Zero: Pixel; (* pixel with all components cleared *)
  102. Bit: ARRAY 100H, 8 OF BOOLEAN; (* Bit[b, i] means bit i in byte b is set *)
  103. Set, Clr: ARRAY 100H, 8 OF CHAR; (* Set/Clr[b, i] is byte b with bit i set/cleared *)
  104. (* d8display: Displays.Display; (* only one system-wide D8 display supported *)
  105. plugin: Plugins.Plugin; *)
  106. MMXenabled*,SSE2enabled* : BOOLEAN;
  107. (**--- Color/Pixel conversions ---**)
  108. (** set pixel to opaque RGB value **)
  109. PROCEDURE SetRGB* (VAR pix: Pixel; red, green, blue: LONGINT);
  110. BEGIN
  111. pix[b] := CHR(blue); pix[g] := CHR(green); pix[r] := CHR(red); pix[a] := 0FFX
  112. END SetRGB;
  113. (** set pixel to partly transparent RGB value **)
  114. PROCEDURE SetRGBA* (VAR pix: Pixel; red, green, blue, alpha: LONGINT);
  115. BEGIN
  116. pix[b] := CHR(blue * alpha DIV 255);
  117. pix[g] := CHR(green * alpha DIV 255);
  118. pix[r] := CHR(red * alpha DIV 255);
  119. pix[a] := CHR(alpha)
  120. END SetRGBA;
  121. (** retrieve RGB and alpha values from pixel **)
  122. PROCEDURE GetRGBA* (pix: Pixel; VAR red, green, blue, alpha: LONGINT);
  123. BEGIN
  124. alpha := ORD(pix[a]);
  125. IF alpha = 0 THEN (* color has disappeared *)
  126. red := 255; green := 255; blue := 255
  127. ELSE
  128. red := ORD(pix[r]); green := ORD(pix[g]); blue := ORD(pix[b]);
  129. IF alpha # 255 THEN (* un-multiply alpha *)
  130. red := 255 * red DIV alpha; IF red > 255 THEN red := 255 END;
  131. green := 255 * green DIV alpha; IF green > 255 THEN green := 255 END;
  132. blue := 255 * blue DIV alpha; IF blue > 255 THEN blue := 255 END
  133. END
  134. END
  135. END GetRGBA;
  136. (**--- Palettes ---**)
  137. (** return index of color in palette which approximates the requested color reasonably well **)
  138. PROCEDURE PaletteIndex* (pal: Palette; red, green, blue: LONGINT): LONGINT;
  139. BEGIN
  140. IF pal.used>256 THEN
  141. RETURN CLUTs.Match(pal.clut, ASH(red, 16) + ASH(green, 8) + blue) *pal.used DIV 256 (*PH090122*)
  142. ELSE RETURN CLUTs.Match(pal.clut, ASH(red, 16) + ASH(green, 8) + blue)
  143. END;
  144. END PaletteIndex;
  145. (** compute internal palette structures whenever palette colors have been modified **)
  146. PROCEDURE InitPalette* (pal: Palette; used, bits: LONGINT);
  147. VAR n, red, green, blue, alpha: LONGINT;
  148. BEGIN
  149. n := 0;
  150. IF used>pal.used THEN NEW(pal.col,used);END;
  151. pal.used:=used;
  152. WHILE n < used DO
  153. GetRGBA(pal.col[n], red, green, blue, alpha);
  154. CLUTs.Set(pal.clut, (*n*) n*255 DIV used, ASH(red, 16) + ASH(green, 8) + blue);
  155. INC(n)
  156. END;
  157. CLUTs.Init(pal.clut, MIN (used,256), bits);
  158. END InitPalette;
  159. (** (re)compute grayscale palette, typically used for pictures with >256 gray scale values**)
  160. PROCEDURE InitGrayPalette* (pal: Palette; used, bits: LONGINT);
  161. VAR n, gray: LONGINT;
  162. BEGIN
  163. n := 0;
  164. IF used>pal.used THEN NEW(pal.col,used); END;
  165. pal.used := used;
  166. WHILE n < used DO
  167. gray:= n*255 DIV used;
  168. SetRGBA(pal.col[n],gray,gray,gray,255); (*PH 090122*)
  169. CLUTs.Set(pal.clut, gray, ASH(gray, 16) + ASH(gray, 8) + gray);
  170. INC(n)
  171. END;
  172. CLUTs.Init(pal.clut, MIN(used,256), bits);
  173. END InitGrayPalette;
  174. (** copy palette contents **)
  175. PROCEDURE CopyPalette* (from, to: Palette);
  176. VAR n: LONGINT;
  177. BEGIN
  178. n := 0;
  179. IF to.used<from.used THEN NEW(to.col,from.used) END;
  180. to.used:=from.used;
  181. WHILE n < from.used DO
  182. to.col[n] := from.col[n]; INC(n)
  183. END;
  184. CLUTs.Copy(from.clut, to.clut)
  185. END CopyPalette;
  186. (** compute and initialize a pseudo-optimal palette for an image (in hi-color or true-color format) **)
  187. PROCEDURE ComputePalette* (img: Image; pal: Palette; reservedcols, maxcols, bits: LONGINT);
  188. (*
  189. uses octree-quantization
  190. *)
  191. TYPE
  192. Node = POINTER TO RECORD (* octree node *)
  193. dsc: ARRAY 8 OF Node; (* descendants *)
  194. link: Node; (* next reducible node on same level *)
  195. leaf: BOOLEAN;
  196. weight: LONGINT; (* accumulated number of pixels represented by this node *)
  197. r, g, b: LONGINT; (* accumulated color values *)
  198. END;
  199. VAR
  200. sent, root: Node; reducible: ARRAY 8 OF Node; colors, maxDepth, y, x, bb, used: LONGINT; adr, aa: ADDRESS; pix: Pixel;
  201. PROCEDURE insert (VAR node: Node; depth: LONGINT; pix: Pixel);
  202. VAR idx, bit: LONGINT;
  203. BEGIN
  204. IF node = NIL THEN
  205. NEW(node);
  206. IF depth = maxDepth THEN
  207. node.leaf := TRUE;
  208. INC(colors)
  209. ELSE
  210. node.leaf := FALSE;
  211. node.link := reducible[depth]; reducible[depth] := node
  212. END
  213. END;
  214. INC(node.weight); (* node represents more pixels *)
  215. IF node.leaf THEN
  216. INC(node.r, LONG(ORD(pix[r])));
  217. INC(node.g, LONG(ORD(pix[g])));
  218. INC(node.b, LONG(ORD(pix[b])))
  219. ELSE
  220. idx := 0; bit := 7-depth;
  221. IF ODD(ASH(ORD(pix[r]), -bit)) THEN INC(idx, 4) END;
  222. IF ODD(ASH(ORD(pix[g]), -bit)) THEN INC(idx, 2) END;
  223. IF ODD(ASH(ORD(pix[b]), -bit)) THEN INC(idx) END;
  224. insert(node.dsc[idx], depth+1, pix)
  225. END
  226. END insert;
  227. PROCEDURE reduce;
  228. VAR d, min, n, i: LONGINT; node, prev, dsc: Node;
  229. BEGIN
  230. d := maxDepth-1;
  231. WHILE reducible[d] = NIL DO
  232. DEC(d); DEC(maxDepth)
  233. END;
  234. sent.link := reducible[d];
  235. node := sent; min := MAX(LONGINT);
  236. WHILE node.link # NIL DO
  237. IF node.link.weight < min THEN
  238. min := node.link.weight; prev := node
  239. END;
  240. node := node.link
  241. END;
  242. node := prev.link; prev.link := node.link;
  243. reducible[d] := sent.link;
  244. n := 1; (* number of colors is initially one for the node itself *)
  245. FOR i := 0 TO 7 DO
  246. dsc := node.dsc[i];
  247. IF dsc # NIL THEN
  248. DEC(n); (* reducing one color *)
  249. INC(node.r, dsc.r); INC(node.g, dsc.g); INC(node.b, dsc.b);
  250. node.dsc[i] := NIL
  251. END
  252. END;
  253. node.leaf := TRUE;
  254. INC(colors, n)
  255. END reduce;
  256. PROCEDURE traverse (node: Node);
  257. VAR i: LONGINT;
  258. BEGIN
  259. IF node # NIL THEN
  260. IF node.leaf THEN
  261. pal.col[used, r] := CHR(node.r DIV node.weight);
  262. pal.col[used, g] := CHR(node.g DIV node.weight);
  263. pal.col[used, b] := CHR(node.b DIV node.weight);
  264. pal.col[used, a] := 0FFX;
  265. INC(used)
  266. ELSE
  267. FOR i := 0 TO 7 DO
  268. traverse(node.dsc[i])
  269. END
  270. END
  271. END
  272. END traverse;
  273. BEGIN (* ComputePalette *)
  274. ASSERT(reservedcols + maxcols <= MAXCOL, 100);
  275. NEW(sent);
  276. root := NIL; colors := 0; maxDepth := 8;
  277. y := 0; adr := img.adr;
  278. WHILE y < img.height DO
  279. IF img.fmt.bpp < 8 THEN
  280. x := 0; aa := adr; bb := 0;
  281. WHILE x < img.width DO
  282. img.fmt.unpack(img.fmt, aa, bb, pix);
  283. insert(root, 0, pix);
  284. WHILE colors > maxcols DO
  285. reduce()
  286. END;
  287. INC(x); bb := bb + img.fmt.bpp; INC(aa, bb DIV 8); bb := bb MOD 8
  288. END
  289. ELSE
  290. x := 0; aa := adr; bb := img.fmt.bpp DIV 8;
  291. WHILE x < img.width DO
  292. img.fmt.unpack(img.fmt, aa, 0, pix);
  293. insert(root, 0, pix);
  294. WHILE colors > maxcols DO
  295. reduce()
  296. END;
  297. INC(x); INC(aa, bb)
  298. END
  299. END;
  300. INC(y); INC(adr, img.bpr)
  301. END;
  302. used := reservedcols;
  303. traverse(root);
  304. InitPalette(pal, used, bits)
  305. END ComputePalette;
  306. (**--- Formats ---**)
  307. (* A1 - one bit alpha, MSB leftmost *)
  308. PROCEDURE PackA1 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  309. VAR b: CHAR;
  310. BEGIN
  311. SYSTEM.GET(adr, b);
  312. IF pix[a] >= 80X THEN SYSTEM.PUT(adr, Set[ORD(b), bit])
  313. ELSE SYSTEM.PUT(adr, Clr[ORD(b), bit])
  314. END
  315. END PackA1;
  316. PROCEDURE UnpackA1 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  317. VAR b: CHAR;
  318. BEGIN
  319. SYSTEM.GET(adr, b);
  320. IF Bit[ORD(b), bit] THEN pix[a] := 0FFX
  321. ELSE pix := Zero
  322. END
  323. END UnpackA1;
  324. (* A8 - 8 bit alpha *)
  325. PROCEDURE PackA8 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  326. BEGIN
  327. SYSTEM.PUT(adr, pix[a])
  328. END PackA8;
  329. PROCEDURE UnpackA8 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  330. VAR alpha: LONGINT;
  331. BEGIN
  332. SYSTEM.GET(adr, pix[a]);
  333. IF pix[a] = 0X THEN
  334. pix := Zero
  335. ELSIF pix[a] # 0FFX THEN
  336. alpha := ORD(pix[a]);
  337. pix[r] := CHR(ORD(pix[r]) * alpha DIV 255);
  338. pix[g] := CHR(ORD(pix[g]) * alpha DIV 255);
  339. pix[b] := CHR(ORD(pix[b]) * alpha DIV 255)
  340. END
  341. END UnpackA8;
  342. (* P8 - 8 bit indexed format with custom palette *)
  343. PROCEDURE PackP8 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  344. BEGIN
  345. IF pix[a] # 0X THEN (* model alpha as brightness *)
  346. SYSTEM.PUT(adr, CHR(CLUTs.Match(fmt.pal.clut, ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))))
  347. END
  348. END PackP8;
  349. PROCEDURE UnpackP8 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  350. VAR idx: CHAR;
  351. BEGIN
  352. SYSTEM.GET(adr, idx); pix := fmt.pal.col[ORD(idx)]
  353. END UnpackP8;
  354. (* D8 - 8 bit indexed format with display palette *)
  355. PROCEDURE PackD8 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  356. BEGIN
  357. IF pix[a] # 0X THEN (* model alpha as brightness *)
  358. SYSTEM.PUT(adr, CHR(ColorToIndex(ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))))
  359. END
  360. END PackD8;
  361. PROCEDURE UnpackD8 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  362. VAR idx: CHAR; col: LONGINT;
  363. BEGIN
  364. SYSTEM.GET(adr, idx); col := IndexToColor(ORD(idx));
  365. pix[b] := CHR(col MOD 100H); pix[g] := CHR(ASH(col, -8) MOD 100H); pix[r] := CHR(ASH(col, -16) MOD 100H)
  366. END UnpackD8;
  367. (* P16 - 16 bit indexed format with custom palette *)
  368. PROCEDURE PackP16 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  369. VAR val:LONGINT;
  370. BEGIN
  371. IF pix[a] # 0X THEN (* model alpha as brightness *)
  372. (*SYSTEM.PUT16(adr, PaletteIndex(fmt.pal, ORD(pix[r]), ORD(pix[g]), ORD(pix[b])))*)
  373. val:=CLUTs.Match(fmt.pal.clut, ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]),16));
  374. IF fmt.pal.used>256 THEN val:=val*fmt.pal.used DIV 256 END;
  375. SYSTEM.PUT16(adr, SHORT(val))
  376. END
  377. END PackP16;
  378. PROCEDURE UnpackP16 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  379. BEGIN
  380. pix := fmt.pal.col[SYSTEM.GET16(adr) MOD 1000H] (*unsigned 16 bit entity*)
  381. END UnpackP16;
  382. (* BGR555 - 16 hi-color with 5 bit blue, 5 bit green and 5 bit red in ascending order *)
  383. PROCEDURE PackBGR555 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  384. VAR int: LONGINT;
  385. BEGIN
  386. IF pix[a] # 0X THEN (* model alpha as brightness *)
  387. int := ASH(ORD(pix[b]), -3) + ASH(ASH(ORD(pix[g]), -3), 5) + ASH(ASH(ORD(pix[r]), -3), 10);
  388. SYSTEM.PUT(adr, CHR(int)); SYSTEM.PUT(adr+1, CHR(ASH(int, -8)))
  389. END
  390. END PackBGR555;
  391. PROCEDURE UnpackBGR555 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  392. VAR lo, hi: CHAR; int: LONGINT;
  393. BEGIN
  394. SYSTEM.GET(adr, lo); SYSTEM.GET(adr+1, hi); int := ASH(ORD(hi), 8) + ORD(lo);
  395. pix[b] := CHR(ASH(int MOD 20H, 3) + 4);
  396. pix[g] := CHR(ASH(ASH(int, -5) MOD 20H, 3) + 4);
  397. pix[r] := CHR(ASH(ASH(int, -10) MOD 20H, 3) + 4);
  398. pix[a] := 0FFX
  399. END UnpackBGR555;
  400. (* BGR565 - 16 hi-color with 5 bit blue, 6 bit green and 5 bit red in ascending order *)
  401. PROCEDURE PackBGR565 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  402. VAR int: LONGINT;
  403. BEGIN
  404. IF pix[a] # 0X THEN (* model alpha as brightness *)
  405. int := ASH(ORD(pix[b]), -3) + ASH(ASH(ORD(pix[g]), -2), 5) + ASH(ASH(ORD(pix[r]), -3), 11);
  406. SYSTEM.PUT(adr, CHR(int)); SYSTEM.PUT(adr+1, CHR(ASH(int, -8)))
  407. END
  408. END PackBGR565;
  409. PROCEDURE UnpackBGR565 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  410. VAR lo, hi: CHAR; int: LONGINT;
  411. BEGIN
  412. SYSTEM.GET(adr, lo); SYSTEM.GET(adr+1, hi); int := ASH(ORD(hi), 8) + ORD(lo);
  413. pix[b] := CHR(ASH(int MOD 20H, 3) + 4);
  414. pix[g] := CHR(ASH(ASH(int, -5) MOD 40H, 2) + 2);
  415. pix[r] := CHR(ASH(ASH(int, -11) MOD 20H, 3) + 4);
  416. pix[a] := 0FFX
  417. END UnpackBGR565;
  418. (* BGR466 - 16 hi-color with 4 bit blue, 6 bit green and 6 bit red in ascending order *)
  419. PROCEDURE PackBGR466 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  420. VAR int: LONGINT;
  421. BEGIN
  422. IF pix[a] # 0X THEN (* model alpha as brightness *)
  423. int := ASH(ORD(pix[b]), -4) + ASH(ASH(ORD(pix[g]), -2), 4) + ASH(ASH(ORD(pix[r]), -2), 10);
  424. SYSTEM.PUT(adr, CHR(int)); SYSTEM.PUT(adr+1, CHR(ASH(int, -8)))
  425. END
  426. END PackBGR466;
  427. PROCEDURE UnpackBGR466 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  428. VAR lo, hi: CHAR; int: LONGINT;
  429. BEGIN
  430. SYSTEM.GET(adr, lo); SYSTEM.GET(adr+1, hi); int := ASH(ORD(hi), 8) + ORD(lo);
  431. pix[b] := CHR(ASH(int MOD 10H, 4) + 8);
  432. pix[g] := CHR(ASH(ASH(int, -4) MOD 40H, 2) + 2);
  433. pix[r] := CHR(ASH(ASH(int, -10) MOD 40H, 2) + 2);
  434. pix[a] := 0FFX
  435. END UnpackBGR466;
  436. (* BGR888 - 24 bit true color with blue in lower, green in middle, and red in upper byte *)
  437. PROCEDURE PackBGR888 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  438. BEGIN
  439. IF pix[a] # 0X THEN (* model alpha as brightness *)
  440. SYSTEM.MOVE(ADDRESSOF(pix[0]), adr, 3)
  441. END
  442. END PackBGR888;
  443. PROCEDURE UnpackBGR888 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  444. BEGIN
  445. SYSTEM.MOVE(adr, ADDRESSOF(pix[0]), 3); pix[a] := 0FFX
  446. END UnpackBGR888;
  447. (* BGRA8888 - 32 bit true color with blue in lowest, green in lower middle, red in upper middle, and alpha in top byte *)
  448. PROCEDURE PackBGRA8888 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  449. BEGIN
  450. SYSTEM.MOVE(ADDRESSOF(pix[0]), adr, 4)
  451. END PackBGRA8888;
  452. PROCEDURE UnpackBGRA8888 (VAR fmt: Format0; adr: ADDRESS; bit: LONGINT; VAR pix: Pixel);
  453. BEGIN
  454. SYSTEM.MOVE(adr, ADDRESSOF(pix[0]), 4)
  455. END UnpackBGRA8888;
  456. (** return image format for given Displays transfer format **)
  457. PROCEDURE DisplayFormat* (format: LONGINT): Format;
  458. BEGIN
  459. CASE format OF
  460. | Displays.index8: RETURN D8
  461. | Displays.color565: RETURN BGR565
  462. | Displays.color888: RETURN BGR888
  463. | Displays.color8888: RETURN BGRA8888
  464. END
  465. END DisplayFormat;
  466. (** initialize format **)
  467. PROCEDURE InitFormat* (VAR fmt: Format; code, bpp, align: SHORTINT; comps: SET; pal: Palette; pack, unpack: PackProc);
  468. BEGIN
  469. fmt.code := code; fmt.bpp := bpp; fmt.align := align; fmt.components := comps; fmt.pal := pal;
  470. fmt.pack := pack; fmt.unpack := unpack
  471. END InitFormat;
  472. (** initialize index formats with custom palette **)
  473. PROCEDURE InitPaletteFormat* (VAR fmt: Format; pal: Palette);
  474. BEGIN
  475. fmt.components := {index}; fmt.pal := pal;
  476. IF pal.used<=256 THEN
  477. fmt.align := 1;
  478. fmt.code := p8; fmt.bpp := 8;
  479. fmt.pack := PackP8; fmt.unpack := UnpackP8
  480. ELSIF pal.used <= 10000H THEN
  481. fmt.align := 2;
  482. fmt.code := p16; fmt.bpp := 16;
  483. fmt.pack := PackP16; fmt.unpack := UnpackP16
  484. ELSE HALT(199)
  485. END
  486. END InitPaletteFormat;
  487. (** return if two formats are the same **)
  488. PROCEDURE Same* (VAR fmt0, fmt1: Format): BOOLEAN;
  489. BEGIN
  490. RETURN
  491. (fmt0.pack = fmt1.pack) & (fmt0.unpack = fmt1.unpack) &
  492. (~(index IN fmt0.components) OR (fmt0.pal = fmt1.pal)) (* doesn't work if palette has been re-initialized *)
  493. END Same;
  494. (**--- Images ---**)
  495. (** initialize custom image **)
  496. PROCEDURE Init* (img: Image; width, height: LONGINT; VAR fmt: Format; bpr: LONGINT; adr: ADDRESS);
  497. BEGIN
  498. ASSERT((width > 0) & (height > 0), 100);
  499. img.width := width; img.height := height; img.fmt := fmt; img.bpr := bpr; img.adr := adr
  500. END Init;
  501. (** initialize custom image on byte buffer **)
  502. PROCEDURE InitBuf* (img: Image; width, height: LONGINT; VAR fmt: Format; bpr, offset: LONGINT; VAR buf: ARRAY OF CHAR);
  503. BEGIN
  504. ASSERT((0 <= offset) & (offset + height * ABS(bpr) <= LEN(buf)), 100);
  505. IF bpr >= 0 THEN Init(img, width, height, fmt, bpr, ADDRESSOF(buf[0]))
  506. ELSE Init(img, width, height, fmt, bpr, ADDRESSOF(buf[offset]) + LEN(buf) - bpr)
  507. END
  508. END InitBuf;
  509. (** initialize image on rectangular area within existing image (lower left corner must fall on byte boundary) **)
  510. PROCEDURE InitRect* (img, base: Image; x, y, w, h: LONGINT);
  511. BEGIN
  512. ASSERT((0 <= x) & (x + w <= base.width) & (0 <= y) & (y + h <= base.height), 100);
  513. ASSERT(x * base.fmt.bpp MOD 8 = 0, 101);
  514. Init(img, w, h, base.fmt, base.bpr, base.adr + y * base.bpr + x * base.fmt.bpp DIV 8)
  515. END InitRect;
  516. (** create image in requested format (allocating or reusing necessary memory) **)
  517. PROCEDURE Create* (img: Image; width, height: LONGINT; fmt: Format);
  518. VAR size: LONGINT; a0, a1: ADDRESS;
  519. BEGIN
  520. ASSERT((width > 0) & (height > 0), 100);
  521. img.width := width; img.height := height;
  522. img.fmt := fmt;
  523. img.bpr := (width * fmt.bpp + 7) DIV 8;
  524. IF fmt.align > 1 THEN
  525. img.bpr := (img.bpr + fmt.align - 1) DIV fmt.align * fmt.align
  526. END;
  527. size := height * img.bpr; INC(size, (-size) MOD 4);
  528. IF (img.mem = NIL) OR (size < LEN(img.mem^) DIV 2) OR (LEN(img.mem^) < size) THEN
  529. NEW(img.mem, size)
  530. ELSE
  531. a0 := ADDRESSOF(img.mem[0]); a1 := a0 + size;
  532. WHILE a0 # a1 DO
  533. SYSTEM.PUT32(a0,0); INC(a0, SIZEOF(LONGINT))
  534. END
  535. END;
  536. img.adr := ADDRESSOF(img.mem[0])
  537. END Create;
  538. PROCEDURE CreateWithBuffer*(img: Image; width, height: LONGINT; fmt: Format; mem: POINTER TO ARRAY OF CHAR; VAR adr: LONGINT);
  539. VAR size: LONGINT; a0, a1: ADDRESS;
  540. BEGIN
  541. ASSERT((width > 0) & (height > 0), 100);
  542. img.width := width; img.height := height;
  543. img.fmt := fmt;
  544. img.bpr := (width * fmt.bpp + 7) DIV 8;
  545. IF fmt.align > 1 THEN
  546. img.bpr := (img.bpr + fmt.align - 1) DIV fmt.align * fmt.align
  547. END;
  548. size := height * img.bpr; INC(size, (-size) MOD 4);
  549. a0 := adr; a1 := adr + size;
  550. ASSERT(ADDRESSOF(mem[0]) <= a0);
  551. ASSERT(a1 <= ADDRESSOF(mem[LEN(mem)-1]));
  552. WHILE a0 # a1 DO
  553. SYSTEM.PUT32(a0,0); INC(a0, SIZEOF(LONGINT))
  554. END;
  555. img.adr := adr;
  556. img.mem := mem;
  557. adr := a1;
  558. END CreateWithBuffer;
  559. (**--- Transfer Modes ---**)
  560. (** initialize transfer mode **)
  561. PROCEDURE InitMode* (VAR mode: Mode; op: SHORTINT);
  562. BEGIN
  563. mode.op := op;
  564. IF (mode.src.pal # NIL) & ((mode.map = NIL) OR (LEN(mode.map^) # mode.src.pal.used)) THEN
  565. NEW(mode.map, mode.src.pal.used)
  566. END;
  567. SetRGB(mode.col, 255, 255, 255);
  568. (*
  569. mode.col := SYSTEM.VAL(Pixel, -1);
  570. *)
  571. mode.src.pack := NIL; mode.dst.pack := NIL (* force re-evaluation of transfer procedure *)
  572. END InitMode;
  573. (** initialize transfer mode with color components for pure alpha sources **)
  574. PROCEDURE InitModeColor* (VAR mode: Mode; op: SHORTINT; red, green, blue: LONGINT);
  575. BEGIN
  576. mode.op := op;
  577. IF (mode.src.pal # NIL) & ((mode.map = NIL) OR (LEN(mode.map^) # mode.src.pal.used)) THEN
  578. NEW(mode.map, mode.src.pal.used)
  579. END;
  580. SetRGB(mode.col, red, green, blue);
  581. mode.src.pack := NIL; mode.dst.pack := NIL
  582. END InitModeColor;
  583. (** set new source color for transfer mode **)
  584. PROCEDURE SetModeColor* (VAR mode: Mode; red, green, blue: LONGINT);
  585. BEGIN
  586. SetRGB(mode.col, red, green, blue);
  587. IF (mode.src.pal # NIL) & ((mode.map = NIL) OR (LEN(mode.map^) # mode.src.pal.used)) THEN
  588. NEW(mode.map, mode.src.pal.used)
  589. END;
  590. mode.src.pack := NIL; mode.dst.pack := NIL
  591. END SetModeColor;
  592. (** blend source pixel into destination pixel according to compositing operation **)
  593. PROCEDURE Blend* (op: LONGINT; VAR src, dst: Pixel);
  594. VAR fs, fd, i: LONGINT;
  595. BEGIN
  596. CASE op OF
  597. | clear: fs := 0; fd := 0
  598. | srcCopy: fs := 255; fd := 0
  599. | dstCopy: fs := 0; fd := 255
  600. | srcOverDst: fs := 255; fd := 255-ORD(src[a])
  601. | dstOverSrc: fs := 255-ORD(dst[a]); fd := 255
  602. | srcInDst: fs := ORD(dst[a]); fd := 0
  603. | dstInSrc: fs := 0; fd := ORD(src[a])
  604. | srcWithoutDst: fs := 255-ORD(dst[a]); fd := 0
  605. | dstWithoutSrc: fs := 0; fd := 255-ORD(src[a])
  606. | srcAtopDst: fs := ORD(dst[a]); fd := 255-ORD(src[a])
  607. | dstAtopSrc: fs := 255-ORD(dst[a]); fd := ORD(src[a])
  608. | srcXorDst: fs := 255-ORD(dst[a]); fd := 255-ORD(src[a])
  609. END;
  610. IF fs + fd = 0 THEN
  611. FOR i := 0 TO 3 DO dst[i] := 0X END
  612. ELSIF fs = 0 THEN
  613. IF fd # 255 THEN
  614. FOR i := 0 TO 3 DO
  615. dst[i] := Clamp[200H + fd * ORD(dst[i]) DIV 255]
  616. END
  617. END
  618. ELSIF fd = 0 THEN
  619. IF fs = 255 THEN
  620. dst := src
  621. ELSE
  622. FOR i := 0 TO 3 DO
  623. dst[i] := Clamp[200H + fs * ORD(src[i]) DIV 255]
  624. END
  625. END
  626. ELSE
  627. dst[0] := Clamp[200H + (fs * ORD(src[0]) + fd * ORD(dst[0])) DIV 255];
  628. dst[1] := Clamp[200H + (fs * ORD(src[1]) + fd * ORD(dst[1])) DIV 255];
  629. dst[2] := Clamp[200H + (fs * ORD(src[2]) + fd * ORD(dst[2])) DIV 255];
  630. dst[3] := Clamp[200H + (fs * ORD(src[3]) + fd * ORD(dst[3])) DIV 255]
  631. END
  632. END Blend;
  633. (*--- General Transfer ---*)
  634. PROCEDURE AnyBlendAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  635. VAR spix, dpix: Pixel;
  636. BEGIN
  637. WHILE len > 0 DO
  638. spix := mode.col; dpix := mode.col;
  639. mode.src.unpack(mode.src, sadr, sbit, spix);
  640. mode.dst.unpack(mode.dst, dadr, dbit, dpix);
  641. Blend(mode.op, spix, dpix);
  642. mode.dst.pack(mode.dst, dadr, dbit, dpix);
  643. sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8;
  644. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8;
  645. DEC(len)
  646. END
  647. END AnyBlendAny;
  648. (* --- invert --- *)
  649. PROCEDURE InvAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  650. BEGIN
  651. WHILE len > 0 DO
  652. mode.dst.unpack(mode.dst, dadr, dbit, mode.buf);
  653. mode.buf[r]:=CHR(255-ORD(mode.buf[r]));
  654. mode.buf[g]:=CHR(255-ORD(mode.buf[g]));
  655. mode.buf[b]:=CHR(255-ORD(mode.buf[b]));
  656. mode.dst.pack(mode.dst, dadr, dbit, mode.buf);
  657. sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8;
  658. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8;
  659. DEC(len)
  660. END
  661. END InvAny;
  662. (* --- alpha invert --- *)
  663. PROCEDURE InvOverAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  664. VAR pix:Pixel;
  665. BEGIN
  666. WHILE len > 0 DO
  667. mode.src.unpack(mode.src, sadr, sbit, pix);
  668. IF pix[a]>=80X THEN
  669. mode.dst.unpack(mode.dst, dadr, dbit, mode.buf);
  670. mode.buf[r]:=CHR(255-ORD(mode.buf[r]));
  671. mode.buf[g]:=CHR(255-ORD(mode.buf[g]));
  672. mode.buf[b]:=CHR(255-ORD(mode.buf[b]));
  673. mode.dst.pack(mode.dst, dadr, dbit, mode.buf)
  674. END;
  675. sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8;
  676. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8;
  677. DEC(len)
  678. END
  679. END InvOverAny;
  680. (*--- clear ---*)
  681. PROCEDURE ClearAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  682. VAR inc: LONGINT;
  683. BEGIN
  684. IF mode.dst.bpp MOD 8 = 0 THEN
  685. inc := mode.dst.bpp DIV 8;
  686. WHILE len > 0 DO
  687. mode.dst.pack(mode.dst, dadr, 0, Zero);
  688. INC(dadr, inc); DEC(len)
  689. END
  690. ELSE
  691. WHILE len > 0 DO
  692. mode.dst.pack(mode.dst, dadr, dbit, Zero);
  693. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8;
  694. DEC(len)
  695. END
  696. END
  697. END ClearAny;
  698. PROCEDURE Clear1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  699. VAR out: CHAR;
  700. BEGIN
  701. IF (dbit > 0) OR (len < 8) THEN
  702. SYSTEM.GET(dadr, out);
  703. WHILE (dbit < 8) & (len > 0) DO
  704. out := Clr[ORD(out), dbit];
  705. INC(dbit); DEC(len)
  706. END;
  707. SYSTEM.PUT(dadr, out)
  708. END;
  709. WHILE len >= 32 DO
  710. SYSTEM.PUT(dadr, LONG(LONG(0))); INC(dadr, 4); DEC(len, 32)
  711. END;
  712. WHILE len >= 8 DO
  713. SYSTEM.PUT(dadr, 0X); INC(dadr); DEC(len, 8)
  714. END;
  715. IF len > 0 THEN
  716. SYSTEM.GET(dadr, out); dbit := 0;
  717. REPEAT
  718. out := Clr[ORD(out), dbit];
  719. INC(dbit); DEC(len)
  720. UNTIL len = 0;
  721. SYSTEM.PUT(dadr, out)
  722. END
  723. END Clear1;
  724. (*
  725. PROCEDURE ClearBytes (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  726. BEGIN
  727. len := len * mode.dst.bpp DIV 8;
  728. WHILE len >= 4 DO
  729. SYSTEM.PUT(dadr, LONG(LONG(0))); INC(dadr, 4); DEC(len)
  730. END;
  731. WHILE len > 0 DO
  732. SYSTEM.PUT(dadr, 0X); INC(dadr); DEC(len)
  733. END
  734. END ClearBytes;
  735. *)
  736. PROCEDURE ClearBytes (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  737. CODE {SYSTEM.i386}
  738. PUSH ECX
  739. MOV EDI, [EBP+dadr]
  740. MOV ECX, [EBP+len]
  741. MOV EAX, 0
  742. CLD
  743. REP STOSB
  744. POP ECX
  745. END ClearBytes;
  746. (* len is nr of DWords*)
  747. PROCEDURE Clear32 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  748. CODE {SYSTEM.i386}
  749. PUSH ECX
  750. MOV EDI, [EBP+dadr]
  751. MOV ECX, [EBP+len]
  752. MOV EAX, 0
  753. CLD
  754. REP STOSD
  755. POP ECX
  756. END Clear32;
  757. (*---- fill --------------*)
  758. PROCEDURE Fill8(destAdr: ADDRESS; size: SIZE; filler: LONGINT);
  759. CODE {SYSTEM.i386}
  760. PUSH ECX
  761. MOV EDI, [EBP+destAdr]
  762. MOV ECX, [EBP+size]
  763. MOV EAX, [EBP+filler]
  764. CLD
  765. REP STOSB
  766. POP ECX
  767. END Fill8;
  768. (* size in 16bit words*)
  769. PROCEDURE Fill16(destAdr: ADDRESS; size: SIZE; filler: LONGINT);
  770. CODE {SYSTEM.i386}
  771. PUSH ECX
  772. MOV EDI, [EBP+destAdr]
  773. MOV ECX, [EBP+size]
  774. MOV EAX, [EBP+filler]
  775. CLD
  776. REP STOSW
  777. POP ECX
  778. END Fill16;
  779. (* size in 24bit entities*)
  780. PROCEDURE Fill24(destAdr: ADDRESS; size: SIZE; filler: LONGINT);
  781. CODE {SYSTEM.i386}
  782. PUSH ECX
  783. MOV EDI, [EBP+destAdr]
  784. IMUL ECX, [EBP+size],3 ; because bytes are used below
  785. MOV EAX, [EBP+filler]
  786. CLD
  787. REP STOSB
  788. POP ECX
  789. END Fill24;
  790. (* size in DWords*)
  791. PROCEDURE Fill32(destAdr: ADDRESS; size: SIZE; filler: LONGINT); (*! to do: change interface, introduce in Blend()*)
  792. CODE {SYSTEM.i386}
  793. PUSH ECX
  794. MOV EDI, [EBP+destAdr]
  795. MOV ECX, [EBP+size]
  796. MOV EAX, [EBP+filler]
  797. CLD
  798. REP STOSD
  799. POP ECX
  800. END Fill32;
  801. (* len in DWords*)
  802. PROCEDURE Fill32A(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT); (*! to do: change interface, introduce in Blend()*)
  803. CODE {SYSTEM.i386}
  804. PUSH ECX
  805. MOV EDI, [EBP+dadr]
  806. MOV ECX, [EBP+len]
  807. MOV EAX, [EBP+sadr]
  808. CLD
  809. REP STOSD
  810. POP ECX
  811. END Fill32A;
  812. (*--- srcCopy Transfer ---*)
  813. (* constant values *)
  814. PROCEDURE Set1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  815. VAR out: CHAR;
  816. BEGIN
  817. IF (dbit > 0) OR (len < 8) THEN
  818. SYSTEM.GET(dadr, out);
  819. WHILE (dbit < 8) & (len > 0) DO
  820. out := Set[ORD(out), dbit];
  821. INC(dbit); DEC(len)
  822. END;
  823. SYSTEM.PUT(dadr, out)
  824. END;
  825. WHILE len >= 8 DO
  826. SYSTEM.PUT(dadr, 0FFX);
  827. INC(dadr); DEC(len, 8)
  828. END;
  829. IF len > 0 THEN
  830. SYSTEM.GET(dadr, out); dbit := 0;
  831. REPEAT
  832. out := Set[ORD(out), dbit];
  833. INC(dbit); DEC(len)
  834. UNTIL len = 0;
  835. SYSTEM.PUT(dadr, out)
  836. END
  837. END Set1;
  838. PROCEDURE ConstCopy8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  839. BEGIN
  840. Fill8(dadr, len, ORD(mode.buf[0]))
  841. (*WHILE len > 0 DO SYSTEM.PUT(dadr, mode.buf[0]); INC(dadr); DEC(len) END*)
  842. END ConstCopy8;
  843. PROCEDURE ConstCopy16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  844. BEGIN
  845. Fill16(dadr, len, ORD(mode.buf[0])+ASH(ORD(mode.buf[1]),8))
  846. (*WHILE len > 0 DO SYSTEM.MOVE(ADDRESSOF(mode.buf[0]), dadr, 2); INC(dadr, 2); DEC(len) END*)
  847. END ConstCopy16;
  848. PROCEDURE ConstCopy24 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  849. BEGIN
  850. Fill24(dadr, len, ORD(mode.buf[0])+ASH(ORD(mode.buf[1]),8)+ASH(ORD(mode.buf[2]),16));
  851. (*WHILE len > 0 DO SYSTEM.MOVE(ADDRESSOF(mode.buf[0]), dadr, 3); INC(dadr, 3); DEC(len) END*)
  852. END ConstCopy24;
  853. (* identical formats *)
  854. PROCEDURE Copy1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  855. VAR in, out: CHAR;
  856. BEGIN
  857. SYSTEM.GET(sadr, in);
  858. WHILE (sbit = 0) & (dbit = 0) & (len >= 8) DO
  859. SYSTEM.PUT(dadr, in);
  860. INC(sadr); INC(dadr); DEC(len, 8);
  861. SYSTEM.GET(sadr, in)
  862. END;
  863. IF (dbit > 0) OR (len < 8) THEN
  864. SYSTEM.GET(dadr, out)
  865. END;
  866. WHILE len > 0 DO
  867. IF Bit[ORD(in), sbit] THEN out := Set[ORD(out), dbit]
  868. ELSE out := Clr[ORD(out), dbit]
  869. END;
  870. INC(sbit); INC(dbit); DEC(len);
  871. IF sbit = 8 THEN
  872. INC(sadr); sbit := 0;
  873. SYSTEM.GET(sadr, in)
  874. END;
  875. IF dbit = 8 THEN
  876. SYSTEM.PUT(dadr, out);
  877. INC(dadr); dbit := 0;
  878. IF len < 8 THEN
  879. SYSTEM.GET(dadr, out)
  880. END
  881. END
  882. END;
  883. IF dbit > 0 THEN
  884. SYSTEM.PUT(dadr, out)
  885. END
  886. END Copy1;
  887. PROCEDURE Copy8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  888. BEGIN
  889. SYSTEM.MOVE(sadr, dadr, len)
  890. END Copy8;
  891. PROCEDURE I8CopyI8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  892. VAR byte: CHAR;
  893. BEGIN
  894. WHILE len > 0 DO
  895. SYSTEM.GET(sadr, byte); SYSTEM.PUT(dadr, mode.map[ORD(byte)]);
  896. INC(sadr); INC(dadr); DEC(len)
  897. END
  898. END I8CopyI8;
  899. PROCEDURE Copy16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  900. BEGIN
  901. SYSTEM.MOVE(sadr, dadr, 2*len)
  902. END Copy16;
  903. (*
  904. PROCEDURE Copy16(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  905. CODE {SYSTEM.i386}
  906. PUSH ECX
  907. MOV ESI, [EBP+sadr]
  908. MOV EDI, [EBP+dadr]
  909. CLD
  910. MOV ECX, [EBP+len]
  911. REP MOVSW
  912. POP ECX
  913. END Copy16;
  914. *)
  915. PROCEDURE I16CopyI16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  916. VAR val: INTEGER;
  917. BEGIN
  918. WHILE len > 0 DO
  919. val:=SYSTEM.GET16(sadr); SYSTEM.PUT16(dadr, mode.map[val MOD 10000H]);
  920. INC(sadr); INC(dadr); DEC(len)
  921. END
  922. END I16CopyI16;
  923. PROCEDURE Copy24 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  924. BEGIN
  925. SYSTEM.MOVE(sadr, dadr, 3*len)
  926. END Copy24;
  927. PROCEDURE Copy32 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  928. BEGIN
  929. SYSTEM.MOVE(sadr, dadr, 4*len)
  930. END Copy32;
  931. (* (*produces only slightly more compact assembler code as prior procedure*)
  932. PROCEDURE Copy32(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  933. CODE {SYSTEM.i386}
  934. PUSH ECX
  935. MOV ESI, [EBP+sadr]
  936. MOV EDI, [EBP+dadr]
  937. CLD
  938. MOV ECX, [EBP+len]
  939. REP MOVSD
  940. POP ECX
  941. END Copy32;
  942. *)
  943. (* general methods *)
  944. PROCEDURE AnyCopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  945. VAR pix: Pixel;
  946. BEGIN
  947. WHILE len > 0 DO
  948. pix := mode.buf;
  949. mode.src.unpack(mode.src, sadr, sbit, mode.buf);
  950. mode.dst.pack(mode.dst, dadr, dbit, mode.buf);
  951. sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8;
  952. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8;
  953. DEC(len)
  954. END
  955. END AnyCopyAny;
  956. PROCEDURE AnyBytesCopyAnyBytes (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  957. VAR sinc, dinc: LONGINT; pix: Pixel;
  958. BEGIN
  959. sinc := mode.src.bpp DIV 8; dinc := mode.dst.bpp DIV 8;
  960. WHILE len > 0 DO
  961. pix := mode.buf;
  962. mode.src.unpack(mode.src, sadr, sbit, mode.buf);
  963. mode.dst.pack(mode.dst, dadr, dbit, mode.buf);
  964. INC(sadr, sinc); INC(dadr, dinc); DEC(len)
  965. END
  966. END AnyBytesCopyAnyBytes;
  967. (* A1 *)
  968. PROCEDURE AnyCopyA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  969. VAR out: CHAR; pix: Pixel;
  970. BEGIN
  971. SYSTEM.GET(dadr, out); pix[a] := 0FFX;
  972. WHILE len > 0 DO
  973. mode.src.unpack(mode.src, sadr, sbit, pix);
  974. sbit := sbit + mode.src.bpp; INC(sadr, sbit MOD 8); sbit := sbit MOD 8;
  975. IF pix[a] >= 80X THEN out := Set[ORD(out), dbit]
  976. ELSE out := Clr[ORD(out), dbit]
  977. END;
  978. INC(dbit); DEC(len);
  979. IF dbit = 8 THEN
  980. SYSTEM.PUT(dadr, out); INC(dadr); SYSTEM.GET(dadr, out); dbit := 0
  981. END
  982. END;
  983. SYSTEM.PUT(dadr, out)
  984. END AnyCopyA1;
  985. PROCEDURE A8CopyA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  986. VAR out, in: CHAR;
  987. BEGIN
  988. IF (dbit > 0) OR (len < 8) THEN
  989. SYSTEM.GET(dadr, out)
  990. END;
  991. WHILE len > 0 DO
  992. SYSTEM.GET(sadr, in);
  993. IF in >= 80X THEN out := Set[ORD(out), dbit]
  994. ELSE out := Clr[ORD(out), dbit]
  995. END;
  996. INC(sadr); INC(dbit); DEC(len);
  997. IF dbit = 8 THEN
  998. SYSTEM.PUT(dadr, out);
  999. INC(dadr); dbit := 0;
  1000. IF len < 8 THEN
  1001. SYSTEM.GET(dadr, out)
  1002. END
  1003. END
  1004. END;
  1005. IF dbit > 0 THEN
  1006. SYSTEM.PUT(dadr, out)
  1007. END
  1008. END A8CopyA1;
  1009. PROCEDURE BGRA8888CopyA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1010. VAR out, in: CHAR;
  1011. BEGIN
  1012. INC(sadr, a); (* only look at alpha component *)
  1013. IF (dbit > 0) OR (len < 8) THEN
  1014. SYSTEM.GET(dadr, out)
  1015. END;
  1016. WHILE len > 0 DO
  1017. SYSTEM.GET(sadr, in);
  1018. IF in >= 80X THEN out := Set[ORD(out), dbit]
  1019. ELSE out := Clr[ORD(out), dbit]
  1020. END;
  1021. INC(sadr, 4); INC(dbit); DEC(len);
  1022. IF dbit = 8 THEN
  1023. SYSTEM.PUT(dadr, out);
  1024. INC(dadr); dbit := 0;
  1025. IF len < 8 THEN
  1026. SYSTEM.GET(dadr, out)
  1027. END
  1028. END
  1029. END;
  1030. IF dbit > 0 THEN
  1031. SYSTEM.PUT(dadr, out)
  1032. END
  1033. END BGRA8888CopyA1;
  1034. PROCEDURE A1CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1035. VAR in: CHAR;
  1036. BEGIN
  1037. SYSTEM.GET(sadr, in);
  1038. WHILE len > 0 DO
  1039. IF Bit[ORD(in), sbit] THEN mode.dst.pack(mode.dst, dadr, dbit, mode.buf)
  1040. ELSE mode.dst.pack(mode.dst, dadr, dbit, Zero)
  1041. END;
  1042. INC(sbit); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len);
  1043. IF sbit = 8 THEN
  1044. INC(sadr); sbit := 0;
  1045. SYSTEM.GET(sadr, in)
  1046. END
  1047. END
  1048. END A1CopyAny;
  1049. PROCEDURE A1CopyA8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1050. VAR in: CHAR;
  1051. BEGIN
  1052. SYSTEM.GET(sadr, in);
  1053. WHILE len > 0 DO
  1054. IF Bit[ORD(in), sbit] THEN SYSTEM.PUT(dadr, 0FFX)
  1055. ELSE SYSTEM.PUT(dadr, 0X)
  1056. END;
  1057. INC(sbit); INC(dadr); DEC(len);
  1058. IF sbit = 8 THEN
  1059. INC(sadr); sbit := 0;
  1060. SYSTEM.GET(sadr, in)
  1061. END
  1062. END
  1063. END A1CopyA8;
  1064. PROCEDURE A1CopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1065. VAR pix: Pixel; in: CHAR;
  1066. BEGIN
  1067. pix := mode.buf;
  1068. SYSTEM.GET(sadr, in);
  1069. WHILE len > 0 DO
  1070. IF Bit[ORD(in), sbit] THEN pix[a] := 0FFX
  1071. ELSE pix[a] := 0X
  1072. END;
  1073. SYSTEM.MOVE(ADDRESSOF(pix), dadr, 4);
  1074. INC(sbit); INC(dadr, 4); DEC(len);
  1075. IF sbit = 8 THEN
  1076. INC(sadr); sbit := 0;
  1077. SYSTEM.GET(sadr, in)
  1078. END
  1079. END
  1080. END A1CopyBGRA8888;
  1081. (* A8 *)
  1082. PROCEDURE AnyCopyA8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1083. VAR pix: Pixel;
  1084. BEGIN
  1085. pix[a] := 0FFX;
  1086. WHILE len > 0 DO
  1087. mode.src.unpack(mode.src, sadr, sbit, pix);
  1088. SYSTEM.PUT(dadr, pix[a]);
  1089. INC(dadr); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1090. END
  1091. END AnyCopyA8;
  1092. PROCEDURE BGRA8888CopyA8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1093. VAR byte: CHAR;
  1094. BEGIN
  1095. INC(sadr, 3);
  1096. WHILE len > 0 DO
  1097. SYSTEM.GET(sadr, byte);
  1098. SYSTEM.PUT(dadr, byte);
  1099. INC(sadr, 4); INC(dadr); DEC(len)
  1100. END
  1101. END BGRA8888CopyA8;
  1102. PROCEDURE A8CopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1103. VAR pix: Pixel;
  1104. BEGIN
  1105. pix := mode.buf;
  1106. WHILE len > 0 DO
  1107. SYSTEM.GET(sadr, pix[a]);
  1108. SYSTEM.MOVE(ADDRESSOF(pix), dadr, 4);
  1109. INC(sadr); INC(dadr, 4); DEC(len)
  1110. END
  1111. END A8CopyBGRA8888;
  1112. (* P8 *)
  1113. PROCEDURE AnyCopyP8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1114. VAR pix: Pixel;
  1115. BEGIN
  1116. WHILE len > 0 DO
  1117. pix := mode.buf;
  1118. mode.src.unpack(mode.src, sadr, sbit, pix);
  1119. SYSTEM.PUT(dadr, CHR(CLUTs.Match(mode.dst.pal.clut, ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1120. INC(dadr); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1121. END
  1122. END AnyCopyP8;
  1123. PROCEDURE Any16CopyP8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1124. VAR pix: Pixel;
  1125. BEGIN
  1126. WHILE len > 0 DO
  1127. mode.src.unpack(mode.src, sadr, 0, pix);
  1128. SYSTEM.PUT(dadr, CHR(CLUTs.Match(mode.dst.pal.clut, ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1129. INC(sadr, 2); INC(dadr); DEC(len)
  1130. END
  1131. END Any16CopyP8;
  1132. PROCEDURE BGR888CopyP8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1133. VAR pix: Pixel;
  1134. BEGIN
  1135. WHILE len > 0 DO
  1136. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 3);
  1137. SYSTEM.PUT(dadr, CHR(CLUTs.Match(mode.dst.pal.clut, ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1138. INC(sadr, 3); INC(dadr); DEC(len)
  1139. END
  1140. END BGR888CopyP8;
  1141. PROCEDURE BGRA8888CopyP8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1142. VAR pix: Pixel;
  1143. BEGIN
  1144. WHILE len > 0 DO
  1145. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 4);
  1146. SYSTEM.PUT(dadr, CHR(CLUTs.Match(mode.dst.pal.clut, ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1147. INC(sadr, 4); INC(dadr); DEC(len)
  1148. END
  1149. END BGRA8888CopyP8;
  1150. PROCEDURE P8CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1151. VAR b: CHAR;
  1152. BEGIN
  1153. WHILE len > 0 DO
  1154. SYSTEM.GET(sadr, b);
  1155. mode.dst.pack(mode.dst, dadr, dbit, mode.src.pal.col[ORD(b)]);
  1156. INC(sadr); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len)
  1157. END
  1158. END P8CopyAny;
  1159. PROCEDURE P8CopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1160. VAR b: CHAR;
  1161. BEGIN
  1162. WHILE len > 0 DO
  1163. SYSTEM.GET(sadr, b);
  1164. mode.dst.pack(mode.dst, dadr, 0, mode.src.pal.col[ORD(b)]);
  1165. INC(sadr); INC(dadr, 2); DEC(len)
  1166. END
  1167. END P8CopyAny16;
  1168. PROCEDURE P8CopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1169. VAR b: CHAR;
  1170. BEGIN
  1171. WHILE len > 0 DO
  1172. SYSTEM.GET(sadr, b);
  1173. SYSTEM.MOVE(ADDRESSOF(mode.src.pal.col[ORD(b)]), dadr, 3);
  1174. INC(sadr); INC(dadr, 3); DEC(len)
  1175. END
  1176. END P8CopyBGR888;
  1177. PROCEDURE P8CopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1178. VAR b: CHAR;
  1179. BEGIN
  1180. WHILE len > 0 DO
  1181. SYSTEM.GET(sadr, b);
  1182. SYSTEM.MOVE(ADDRESSOF(mode.src.pal.col[ORD(b)]), dadr, 4);
  1183. INC(sadr); INC(dadr, 4); DEC(len)
  1184. END
  1185. END P8CopyBGRA8888;
  1186. (* D8 *)
  1187. PROCEDURE AnyCopyD8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1188. VAR pix: Pixel;
  1189. BEGIN
  1190. WHILE len > 0 DO
  1191. pix := mode.buf;
  1192. mode.src.unpack(mode.src, sadr, sbit, pix);
  1193. SYSTEM.PUT(dadr, CHR(ColorToIndex(ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1194. INC(dadr); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1195. END
  1196. END AnyCopyD8;
  1197. PROCEDURE Any16CopyD8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1198. VAR pix: Pixel;
  1199. BEGIN
  1200. WHILE len > 0 DO
  1201. mode.src.unpack(mode.src, sadr, 0, pix);
  1202. SYSTEM.PUT(dadr, CHR(ColorToIndex(ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1203. INC(sadr, 2); INC(dadr); DEC(len)
  1204. END
  1205. END Any16CopyD8;
  1206. PROCEDURE BGR888CopyD8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1207. VAR pix: Pixel;
  1208. BEGIN
  1209. WHILE len > 0 DO
  1210. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 3);
  1211. SYSTEM.PUT(dadr, CHR(ColorToIndex(ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1212. INC(sadr, 3); INC(dadr); DEC(len)
  1213. END
  1214. END BGR888CopyD8;
  1215. PROCEDURE BGRA8888CopyD8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1216. VAR pix: Pixel;
  1217. BEGIN
  1218. WHILE len > 0 DO
  1219. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 4);
  1220. SYSTEM.PUT(dadr, CHR(ColorToIndex(ORD(pix[b]) + ASH(ORD(pix[g]), 8) + ASH(ORD(pix[r]), 16))));
  1221. INC(sadr, 4); INC(dadr); DEC(len)
  1222. END
  1223. END BGRA8888CopyD8;
  1224. PROCEDURE D8CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1225. VAR pix: Pixel; byte: CHAR; col: LONGINT;
  1226. BEGIN
  1227. pix[a] := 0FFX;
  1228. WHILE len > 0 DO
  1229. SYSTEM.GET(sadr, byte); col := IndexToColor(ORD(byte));
  1230. pix[b] := CHR(col MOD 100H); pix[g] := CHR(ASH(col, -8) MOD 100H); pix[r] := CHR(ASH(col, -16) MOD 100H);
  1231. mode.dst.pack(mode.dst, dadr, dbit, pix);
  1232. INC(sadr); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len)
  1233. END
  1234. END D8CopyAny;
  1235. PROCEDURE D8CopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1236. VAR pix: Pixel; byte: CHAR; col: LONGINT;
  1237. BEGIN
  1238. pix[a] := 0FFX;
  1239. WHILE len > 0 DO
  1240. SYSTEM.GET(sadr, byte); col := IndexToColor(ORD(byte));
  1241. pix[b] := CHR(col MOD 100H); pix[g] := CHR(ASH(col, -8) MOD 100H); pix[r] := CHR(ASH(col, -16) MOD 100H);
  1242. mode.dst.pack(mode.dst, dadr, 0, pix);
  1243. INC(sadr); INC(dadr, 2); DEC(len)
  1244. END
  1245. END D8CopyAny16;
  1246. PROCEDURE D8CopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1247. VAR byte: CHAR; col: LONGINT; pix: Pixel;
  1248. BEGIN
  1249. WHILE len > 0 DO
  1250. SYSTEM.GET(sadr, byte); col := IndexToColor(ORD(byte));
  1251. pix[b] := CHR(col MOD 100H); pix[g] := CHR(ASH(col, -8) MOD 100H); pix[r] := CHR(ASH(col, -16) MOD 100H);
  1252. SYSTEM.MOVE(ADDRESSOF(pix[0]), dadr, 3);
  1253. INC(sadr); INC(dadr, 3); DEC(len)
  1254. END
  1255. END D8CopyBGR888;
  1256. PROCEDURE D8CopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1257. VAR pix: Pixel; byte: CHAR; col: LONGINT;
  1258. BEGIN
  1259. pix[a] := 0FFX;
  1260. WHILE len > 0 DO
  1261. SYSTEM.GET(sadr, byte); col := IndexToColor(ORD(byte));
  1262. pix[b] := CHR(col MOD 100H); pix[g] := CHR(ASH(col, -8) MOD 100H); pix[r] := CHR(ASH(col, -16) MOD 100H);
  1263. SYSTEM.MOVE(ADDRESSOF(pix[0]), dadr, 4);
  1264. INC(sadr); INC(dadr, 4); DEC(len)
  1265. END
  1266. END D8CopyBGRA8888;
  1267. (*
  1268. (* P816*)
  1269. PROCEDURE AnyCopyP16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1270. VAR pix: Pixel;
  1271. BEGIN
  1272. WHILE len > 0 DO
  1273. pix := mode.buf;
  1274. mode.src.unpack(mode.src, sadr, sbit, pix);
  1275. SYSTEM.PUT16(dadr, PaletteIndex(mode.dst.pal, ORD(pix[r]), ORD(pix[g]), ORD(pix[b])));
  1276. INC(dadr,2); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1277. END
  1278. END AnyCopyP16;
  1279. PROCEDURE Any16CopyP16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1280. VAR pix: Pixel;
  1281. BEGIN
  1282. WHILE len > 0 DO
  1283. mode.src.unpack(mode.src, sadr, 0, pix);
  1284. SYSTEM.PUT16(dadr, PaletteIndex(mode.dst.pal, ORD(pix[r]), ORD(pix[g]), ORD(pix[b])));
  1285. INC(sadr, 2); INC(dadr,2); DEC(len)
  1286. END
  1287. END Any16CopyP16;
  1288. PROCEDURE BGR888CopyP16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1289. VAR pix: Pixel;
  1290. BEGIN
  1291. WHILE len > 0 DO
  1292. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 3);
  1293. SYSTEM.PUT16(dadr, PaletteIndex(mode.dst.pal, ORD(pix[r]), ORD(pix[g]), ORD(pix[b])));
  1294. INC(sadr, 3); INC(dadr,2); DEC(len)
  1295. END
  1296. END BGR888CopyP16;
  1297. PROCEDURE BGRA8888CopyP16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1298. VAR pix: Pixel;
  1299. BEGIN
  1300. WHILE len > 0 DO
  1301. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 4);
  1302. SYSTEM.PUT16(dadr, PaletteIndex(mode.dst.pal, ORD(pix[r]), ORD(pix[g]), ORD(pix[b])));
  1303. INC(sadr, 4); INC(dadr,2); DEC(len)
  1304. END
  1305. END BGRA8888CopyP16;
  1306. PROCEDURE P16CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1307. BEGIN
  1308. WHILE len > 0 DO
  1309. mode.dst.pack(mode.dst, dadr, dbit, mode.src.pal.col[LONG(SYSTEM.GET16(sadr)) MOD 10000H]);
  1310. INC(sadr,2); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len)
  1311. END
  1312. END P16CopyAny;
  1313. PROCEDURE P16CopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1314. BEGIN
  1315. WHILE len > 0 DO
  1316. mode.dst.pack(mode.dst, dadr, 0, mode.src.pal.col[LONG(SYSTEM.GET16(sadr)) MOD 10000H]);
  1317. INC(sadr,2); INC(dadr, 2); DEC(len)
  1318. END
  1319. END P16CopyAny16;
  1320. *)
  1321. PROCEDURE P16CopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1322. BEGIN
  1323. WHILE len > 0 DO
  1324. SYSTEM.MOVE(ADDRESSOF(mode.src.pal.col[LONG(SYSTEM.GET16(sadr)) MOD 10000H]), dadr, 3);
  1325. INC(sadr,2); INC(dadr, 3); DEC(len)
  1326. END
  1327. END P16CopyBGR888;
  1328. PROCEDURE P16CopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1329. VAR val:LONGINT;
  1330. BEGIN
  1331. WHILE len > 0 DO
  1332. val:=LONG(SYSTEM.GET16(sadr)) MOD 10000H;
  1333. SYSTEM.MOVE(ADDRESSOF(mode.src.pal.col[val]), dadr, 4);
  1334. INC(sadr,2); INC(dadr, 4); DEC(len)
  1335. END
  1336. END P16CopyBGRA8888;
  1337. (* BGR555, BGR565, BGR466 *)
  1338. PROCEDURE AnyCopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1339. VAR pix: Pixel;
  1340. BEGIN
  1341. WHILE len > 0 DO
  1342. pix := mode.buf;
  1343. mode.src.unpack(mode.src, sadr, 0, pix);
  1344. mode.dst.pack(mode.dst, dadr, 0, pix);
  1345. INC(dadr, 2); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1346. END
  1347. END AnyCopyAny16;
  1348. PROCEDURE Any16CopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1349. VAR pix: Pixel;
  1350. BEGIN
  1351. WHILE len > 0 DO
  1352. mode.src.unpack(mode.src, sadr, 0, pix);
  1353. mode.dst.pack(mode.dst, dadr, 0, pix);
  1354. INC(sadr, 2); INC(dadr, 2); DEC(len)
  1355. END
  1356. END Any16CopyAny16;
  1357. PROCEDURE BGR888CopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1358. VAR pix: Pixel;
  1359. BEGIN
  1360. pix[a] := 0FFX;
  1361. WHILE len > 0 DO
  1362. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 3);
  1363. mode.dst.pack(mode.dst, dadr, 0, pix);
  1364. INC(sadr, 3); INC(dadr, 2); DEC(len)
  1365. END
  1366. END BGR888CopyAny16;
  1367. PROCEDURE BGRA8888CopyAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1368. VAR pix: Pixel;
  1369. BEGIN
  1370. WHILE len > 0 DO
  1371. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 4);
  1372. mode.dst.pack(mode.dst, dadr, 0, pix);
  1373. INC(dadr, 2); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1374. END
  1375. END BGRA8888CopyAny16;
  1376. PROCEDURE Any16CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1377. VAR pix: Pixel;
  1378. BEGIN
  1379. WHILE len > 0 DO
  1380. mode.src.unpack(mode.src, sadr, 0, pix);
  1381. mode.dst.pack(mode.dst, dadr, 0, pix);
  1382. INC(sadr, 2); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len)
  1383. END
  1384. END Any16CopyAny;
  1385. PROCEDURE Any16CopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1386. VAR pix: Pixel;
  1387. BEGIN
  1388. WHILE len > 0 DO
  1389. mode.src.unpack(mode.src, sadr, 0, pix);
  1390. SYSTEM.MOVE(ADDRESSOF(pix[0]), dadr, 3);
  1391. INC(sadr, 2); INC(dadr, 3); DEC(len)
  1392. END
  1393. END Any16CopyBGR888;
  1394. PROCEDURE Any16CopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1395. VAR pix: Pixel;
  1396. BEGIN
  1397. WHILE len > 0 DO
  1398. mode.src.unpack(mode.src, sadr, 0, pix);
  1399. SYSTEM.MOVE(ADDRESSOF(pix[0]), dadr, 4);
  1400. INC(sadr, 2); INC(dadr, 4); DEC(len)
  1401. END
  1402. END Any16CopyBGRA8888;
  1403. (* BGR888 *)
  1404. PROCEDURE AnyCopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1405. VAR pix: Pixel;
  1406. BEGIN
  1407. WHILE len > 0 DO
  1408. pix := mode.buf;
  1409. mode.src.unpack(mode.src, sadr, sbit, pix);
  1410. SYSTEM.MOVE(ADDRESSOF(pix[0]), dadr, 3);
  1411. INC(dadr, 3); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1412. END
  1413. END AnyCopyBGR888;
  1414. PROCEDURE BGRA8888CopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1415. BEGIN
  1416. WHILE len > 0 DO
  1417. SYSTEM.MOVE(sadr, dadr, 3);
  1418. INC(sadr, 4); INC(dadr, 3); DEC(len)
  1419. END
  1420. END BGRA8888CopyBGR888;
  1421. PROCEDURE SSE2BGRA8888CopyBGR888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1422. CODE {SYSTEM.i386, SYSTEM.MMX, SYSTEM.SSE, SYSTEM.SSE2, SYSTEM.Pentium}
  1423. PUSHFD
  1424. PUSH ECX
  1425. PUSH EBX
  1426. ; CLI
  1427. MOV ESI, [EBP+sadr] ; source adr
  1428. MOV EDI, [EBP+dadr] ; source adr
  1429. MOV ECX, [EBP+len]
  1430. loop:
  1431. CMP ECX, 0
  1432. JLE end
  1433. CMP ECX, 4
  1434. JL singlepixel
  1435. fourpixel:
  1436. ; 4pixels at the time
  1437. MOV EAX, [ESI] ; pixel 0
  1438. MOV EBX, [ESI+4] ; pixel 1
  1439. AND EAX, 0FFFFFFH
  1440. AND EBX, 0FFFFFFH
  1441. MOV EDX, EBX
  1442. SHL EDX, 24
  1443. OR EAX, EDX ; 1000
  1444. MOV [EDI], EAX ; write back to mem
  1445. MOV EAX, [ESI+8] ; pixel 2
  1446. AND EAX, 0FFFFFFH
  1447. SHR EBX,8
  1448. MOV EDX, EAX
  1449. SHL EDX, 16
  1450. OR EBX, EDX ; 2211
  1451. MOV [EDI+4], EBX
  1452. MOV EDX, [ESI+12] ; pixel 3
  1453. SHL EDX, 8
  1454. SHR EAX, 16
  1455. OR EAX, EDX ; 3332
  1456. MOV [EDI], EAX
  1457. ADD ESI, 16
  1458. ADD EDI, 12
  1459. SUB ECX, 4
  1460. JG loop
  1461. JMP end
  1462. singlepixel:
  1463. MOV EAX, [ESI]
  1464. MOV [EDI], AX
  1465. SHR EAX, 16
  1466. MOV [EDI+2], AL
  1467. ADD ESI, 4
  1468. ADD EDI, 3
  1469. SUB ECX, 1
  1470. JG loop
  1471. end:
  1472. EMMS ; declare FPU registers free
  1473. POP EBX
  1474. POP ECX
  1475. POPFD
  1476. END SSE2BGRA8888CopyBGR888;
  1477. PROCEDURE BGR888CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1478. VAR pix: Pixel;
  1479. BEGIN
  1480. pix[a] := 0FFX;
  1481. WHILE len > 0 DO
  1482. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 3);
  1483. mode.dst.pack(mode.dst, dadr, dbit, pix);
  1484. INC(sadr, 3); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len)
  1485. END
  1486. END BGR888CopyAny;
  1487. PROCEDURE BGR888CopyBGRA8888(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1488. BEGIN
  1489. WHILE len > 0 DO
  1490. SYSTEM.MOVE(sadr, dadr, 3); SYSTEM.PUT(dadr+3, 0FFX);
  1491. INC(sadr, 3); INC(dadr, 4); DEC(len)
  1492. END
  1493. END BGR888CopyBGRA8888;
  1494. PROCEDURE SSE2BGR888CopyBGRA8888(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1495. CODE {SYSTEM.i386, SYSTEM.MMX, SYSTEM.SSE, SYSTEM.SSE2,SYSTEM.Pentium}
  1496. PUSHFD
  1497. PUSH ECX
  1498. PUSH EBX
  1499. ; CLI
  1500. PXOR XMM0, XMM0
  1501. MOV EAX, 0FF00H
  1502. PINSRW XMM0, EAX, 1
  1503. PINSRW XMM0, EAX, 3
  1504. PINSRW XMM0, EAX, 5
  1505. PINSRW XMM0, EAX, 7 ; prepare for fourpixel
  1506. SHL EAX, 16 ; prepare for singlepixel
  1507. MOV ESI, [EBP+sadr] ; source adr
  1508. MOV EDI, [EBP+dadr] ; source adr
  1509. MOV ECX, [EBP+len]
  1510. loop:
  1511. CMP ECX, 0
  1512. JLE end
  1513. CMP ECX, 4
  1514. JL singlepixel
  1515. fourpixel:
  1516. ; 4pixels at the time
  1517. PXOR XMM2,XMM2
  1518. PXOR XMM1,XMM1
  1519. MOV EBX, [ESI+9] ; read 1st source pixel
  1520. MOVD XMM2, EBX
  1521. PSLLDQ XMM2, 4
  1522. MOV EBX, [ESI+6] ; read 2nd source pixel
  1523. MOVD XMM1, EBX
  1524. POR XMM2, XMM1
  1525. PSLLDQ XMM2, 4
  1526. MOV EBX, [ESI+3] ; read 3rd source pixel
  1527. MOVD XMM1, EBX
  1528. POR XMM2, XMM1
  1529. PSLLDQ XMM2, 4
  1530. MOV EBX, [ESI] ; read 4th source pixel
  1531. MOVD XMM1, EBX
  1532. POR XMM2, XMM1
  1533. ADD ESI, 12
  1534. POR XMM2, XMM0
  1535. MOVDQU [EDI], XMM2 ; set the pixels
  1536. ADD EDI, 16 ; inc adr
  1537. SUB ECX, 4
  1538. JG loop
  1539. JMP end
  1540. singlepixel:
  1541. MOV EBX, [ESI] ; read source pixel
  1542. OR EBX, EAX
  1543. ADD ESI, 3
  1544. MOV [EDI], EBX
  1545. ADD EDI, 4 ; inc adr
  1546. SUB ECX, 1
  1547. JG loop
  1548. end:
  1549. EMMS ; declare FPU registers free
  1550. POP EBX
  1551. POP ECX
  1552. POPFD
  1553. END SSE2BGR888CopyBGRA8888;
  1554. (* BGRA8888 *)
  1555. PROCEDURE AnyCopyBGRA8888 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1556. VAR pix: Pixel;
  1557. BEGIN
  1558. WHILE len > 0 DO
  1559. pix := mode.buf;
  1560. mode.src.unpack(mode.src, sadr, sbit, pix);
  1561. SYSTEM.MOVE(ADDRESSOF(pix[0]), dadr, 4);
  1562. INC(dadr, 4); sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; DEC(len)
  1563. END
  1564. END AnyCopyBGRA8888;
  1565. PROCEDURE BGRA8888CopyAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1566. VAR pix: Pixel;
  1567. BEGIN
  1568. WHILE len > 0 DO
  1569. SYSTEM.MOVE(sadr, ADDRESSOF(pix[0]), 4);
  1570. mode.dst.pack(mode.dst, dadr, dbit, pix);
  1571. INC(sadr, 4); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len)
  1572. END
  1573. END BGRA8888CopyAny;
  1574. (*--- dstCopy Transfer ---*)
  1575. PROCEDURE EmptyTransfer (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1576. END EmptyTransfer;
  1577. (*--- srcOverDst Transfer ---*)
  1578. (* A1 *)
  1579. PROCEDURE AnyOverA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1580. VAR out: CHAR; pix: Pixel;
  1581. BEGIN
  1582. SYSTEM.GET(dadr, out);
  1583. WHILE len > 0 DO
  1584. mode.src.unpack(mode.src, sadr, sbit, pix);
  1585. IF pix[a] >= 80X THEN
  1586. out := Set[ORD(out), dbit]
  1587. END;
  1588. sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; INC(dbit); DEC(len);
  1589. IF dbit = 8 THEN
  1590. SYSTEM.PUT(dadr, out);
  1591. INC(dadr); dbit := 0;
  1592. SYSTEM.GET(dadr, out)
  1593. END
  1594. END;
  1595. SYSTEM.PUT(dadr, out)
  1596. END AnyOverA1;
  1597. PROCEDURE A1OverA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1598. VAR in, out: CHAR;
  1599. BEGIN
  1600. SYSTEM.GET(sadr, in); SYSTEM.GET(dadr, out);
  1601. WHILE len > 0 DO
  1602. IF Bit[ORD(in), sbit] THEN
  1603. out := Set[ORD(out), dbit]
  1604. END;
  1605. INC(sbit); INC(dbit); DEC(len);
  1606. IF sbit = 8 THEN
  1607. INC(sadr); sbit := 0;
  1608. SYSTEM.GET(sadr, in)
  1609. END;
  1610. IF dbit = 8 THEN
  1611. SYSTEM.PUT(dadr, out);
  1612. INC(dadr); dbit := 0;
  1613. SYSTEM.GET(dadr, out)
  1614. END
  1615. END;
  1616. SYSTEM.PUT(dadr, out)
  1617. END A1OverA1;
  1618. PROCEDURE A8OverA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1619. VAR in, out: CHAR;
  1620. BEGIN
  1621. SYSTEM.GET(dadr, out);
  1622. WHILE len > 0 DO
  1623. SYSTEM.GET(sadr, in);
  1624. IF in >= 80X THEN
  1625. out := Set[ORD(out), dbit]
  1626. END;
  1627. INC(sadr); INC(dbit); DEC(len);
  1628. IF dbit = 8 THEN
  1629. SYSTEM.PUT(dadr, out);
  1630. INC(dadr); dbit := 0;
  1631. SYSTEM.GET(dadr, out)
  1632. END
  1633. END;
  1634. SYSTEM.PUT(dadr, out)
  1635. END A8OverA1;
  1636. PROCEDURE BGRA8888OverA1 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1637. VAR in, out: CHAR;
  1638. BEGIN
  1639. SYSTEM.GET(dadr, out);
  1640. WHILE len > 0 DO
  1641. SYSTEM.GET(sadr + a, in);
  1642. IF in >= 80X THEN
  1643. out := Set[ORD(out), dbit]
  1644. END;
  1645. INC(sadr, 4); INC(dbit); DEC(len);
  1646. IF dbit = 8 THEN
  1647. SYSTEM.PUT(dadr, out);
  1648. INC(dadr); dbit := 0;
  1649. SYSTEM.GET(dadr, out)
  1650. END
  1651. END;
  1652. SYSTEM.PUT(dadr, out)
  1653. END BGRA8888OverA1;
  1654. PROCEDURE A1OverAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1655. VAR in: CHAR;
  1656. BEGIN
  1657. SYSTEM.GET(sadr, in);
  1658. WHILE len > 0 DO
  1659. IF Bit[ORD(in), sbit] THEN
  1660. mode.dst.pack(mode.dst, dadr, dbit, mode.buf)
  1661. END;
  1662. INC(sbit); dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8; DEC(len);
  1663. IF sbit = 8 THEN
  1664. INC(sadr); sbit := 0;
  1665. SYSTEM.GET(sadr, in)
  1666. END
  1667. END
  1668. END A1OverAny;
  1669. PROCEDURE A1OverConst8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1670. VAR in: CHAR;
  1671. BEGIN
  1672. SYSTEM.GET(sadr, in);
  1673. WHILE len > 0 DO
  1674. IF Bit[ORD(in), sbit] THEN
  1675. SYSTEM.PUT(dadr, mode.buf[0])
  1676. END;
  1677. INC(sbit); INC(dadr); DEC(len);
  1678. IF sbit = 8 THEN
  1679. INC(sadr); sbit := 0;
  1680. SYSTEM.GET(sadr, in)
  1681. END
  1682. END
  1683. END A1OverConst8;
  1684. PROCEDURE A1OverConst16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1685. VAR in: CHAR;
  1686. BEGIN
  1687. SYSTEM.GET(sadr, in);
  1688. WHILE len > 0 DO
  1689. IF Bit[ORD(in), sbit] THEN
  1690. SYSTEM.MOVE(ADDRESSOF(mode.buf[0]), dadr, 2)
  1691. END;
  1692. INC(sbit); INC(dadr, 2); DEC(len);
  1693. IF sbit = 8 THEN
  1694. INC(sadr); sbit := 0;
  1695. SYSTEM.GET(sadr, in)
  1696. END
  1697. END
  1698. END A1OverConst16;
  1699. PROCEDURE A1OverConst24 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1700. VAR in: CHAR;
  1701. BEGIN
  1702. SYSTEM.GET(sadr, in);
  1703. WHILE len > 0 DO
  1704. IF Bit[ORD(in), sbit] THEN
  1705. SYSTEM.MOVE(ADDRESSOF(mode.buf[0]), dadr, 3)
  1706. END;
  1707. INC(sbit); INC(dadr, 3); DEC(len);
  1708. IF sbit = 8 THEN
  1709. INC(sadr); sbit := 0;
  1710. SYSTEM.GET(sadr, in)
  1711. END
  1712. END
  1713. END A1OverConst24;
  1714. PROCEDURE A1OverConst32 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1715. VAR in: CHAR;
  1716. BEGIN
  1717. SYSTEM.GET(sadr, in);
  1718. WHILE len > 0 DO
  1719. IF Bit[ORD(in), sbit] THEN
  1720. SYSTEM.MOVE(ADDRESSOF(mode.buf[0]), dadr, 4)
  1721. END;
  1722. INC(sbit); INC(dadr, 4); DEC(len);
  1723. IF sbit = 8 THEN
  1724. INC(sadr); sbit := 0;
  1725. SYSTEM.GET(sadr, in)
  1726. END
  1727. END
  1728. END A1OverConst32;
  1729. (* A8 *)
  1730. PROCEDURE AnyOverA8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1731. VAR pix: Pixel; b: CHAR;
  1732. BEGIN
  1733. WHILE len > 0 DO
  1734. mode.src.unpack(mode.src, sadr, sbit, pix);
  1735. IF pix[a] = 0FFX THEN
  1736. SYSTEM.PUT(dadr, 0FFX)
  1737. ELSIF pix[a] # 0X THEN
  1738. SYSTEM.GET(dadr, b);
  1739. SYSTEM.PUT(dadr, CHR(ORD(pix[a]) + ORD(b) * LONG(255-ORD(pix[a])) DIV 255))
  1740. END;
  1741. sbit := sbit + mode.src.bpp; INC(sadr, sbit DIV 8); sbit := sbit MOD 8; INC(dadr); DEC(len)
  1742. END
  1743. END AnyOverA8;
  1744. PROCEDURE A8OverAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1745. VAR spix, dpix: Pixel; alpha: CHAR;
  1746. BEGIN
  1747. WHILE len > 0 DO
  1748. SYSTEM.GET(sadr, alpha);
  1749. IF alpha = 0FFX THEN
  1750. mode.dst.pack(mode.dst, dadr, dbit, mode.buf)
  1751. ELSIF mode.buf[a] # 0X THEN
  1752. spix[a] := alpha;
  1753. (* the following computation of the colors has to be done because the blending method seems to assume this *)
  1754. spix[r] := CHR(ORD(mode.col[r]) * ORD(alpha) DIV 255);
  1755. spix[g] := CHR(ORD(mode.col[g]) * ORD(alpha) DIV 255);
  1756. spix[b] := CHR(ORD(mode.col[b]) * ORD(alpha) DIV 255);
  1757. mode.dst.unpack(mode.dst, dadr, dbit, dpix);
  1758. Blend(mode.op, spix, dpix);
  1759. mode.dst.pack(mode.dst, dadr, dbit, dpix);
  1760. END;
  1761. INC(sadr); DEC(len);
  1762. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8
  1763. END
  1764. END A8OverAny;
  1765. PROCEDURE A8OverBGRA8888(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1766. VAR src , dst: Pixel; rc,gc,bc, fd, fs: LONGINT;
  1767. BEGIN
  1768. rc := ORD(mode.col[r]);
  1769. gc := ORD(mode.col[g]);
  1770. bc := ORD(mode.col[b]);
  1771. fs := 255;
  1772. WHILE len > 0 DO
  1773. SYSTEM.GET(sadr, src[a]);
  1774. SYSTEM.MOVE(dadr, ADDRESSOF(dst), 4);
  1775. (* the following computation of the colors has to be done because the blending method seems to assume this *)
  1776. src[r] := CHR(rc * ORD(src[a]) DIV 255);
  1777. src[g] := CHR(gc * ORD(src[a]) DIV 255);
  1778. src[b] := CHR(bc * ORD(src[a]) DIV 255);
  1779. fd := 255-ORD(src[a]);
  1780. dst[0] := Clamp[200H + (fs * ORD(src[0]) + fd * ORD(dst[0])) DIV 255];
  1781. dst[1] := Clamp[200H + (fs * ORD(src[1]) + fd * ORD(dst[1])) DIV 255];
  1782. dst[2] := Clamp[200H + (fs * ORD(src[2]) + fd * ORD(dst[2])) DIV 255];
  1783. dst[3] := Clamp[200H + (fs * ORD(src[3]) + fd * ORD(dst[3])) DIV 255];
  1784. SYSTEM.MOVE(ADDRESSOF(dst),dadr,4);
  1785. INC(sadr); INC(dadr,4); DEC(len);
  1786. END
  1787. END A8OverBGRA8888;
  1788. PROCEDURE A8OverA8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1789. VAR in, out: CHAR;
  1790. BEGIN
  1791. WHILE len > 0 DO
  1792. SYSTEM.GET(sadr, in);
  1793. IF in = 0FFX THEN
  1794. SYSTEM.PUT(dadr, 0FFX)
  1795. ELSIF in # 0X THEN
  1796. SYSTEM.GET(dadr, out);
  1797. SYSTEM.PUT(dadr, CHR(ORD(in) + ORD(out) * LONG(255-ORD(in)) DIV 255))
  1798. END;
  1799. INC(sadr); INC(dadr); DEC(len)
  1800. END
  1801. END A8OverA8;
  1802. PROCEDURE BGRA8888OverBGRA8888(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1803. VAR src, dst: Pixel; fd,t: LONGINT;
  1804. BEGIN
  1805. WHILE len > 0 DO
  1806. SYSTEM.MOVE(sadr, ADDRESSOF(src), 4);
  1807. IF src[a] = 0FFX THEN
  1808. SYSTEM.MOVE(sadr,dadr,4);
  1809. ELSIF src[a] # 0X THEN
  1810. SYSTEM.MOVE(dadr, ADDRESSOF(dst), 4);
  1811. fd := 256-ORD(src[a]);
  1812. dst[0] := CHR(MIN( (256 * ORD(src[0]) + fd * ORD(dst[0])) DIV 256, 255));
  1813. (*
  1814. IF t < 255 THEN dst[0] := CHR(t) ELSE dst[0] := CHR(255) END;
  1815. *)
  1816. dst[1] := CHR(MIN((256 * ORD(src[1]) + fd * ORD(dst[1])) DIV 256, 255));
  1817. (*
  1818. IF t < 255 THEN dst[1] := CHR(t) ELSE dst[1] := CHR(255) END;
  1819. *)
  1820. dst[2] := CHR(MIN( (256 * ORD(src[2]) + fd * ORD(dst[2])) DIV 256, 255));
  1821. (*
  1822. IF t < 255 THEN dst[2] := CHR(t) ELSE dst[2] := CHR(255) END;
  1823. *)
  1824. dst[3] := CHR(MIN( (256 * ORD(src[3]) + fd * ORD(dst[3])) DIV 256, 255));
  1825. (*
  1826. IF t < 255 THEN dst[3] := CHR(t) ELSE dst[3] := CHR(255) END;
  1827. *)
  1828. SYSTEM.MOVE(ADDRESSOF(dst),dadr,4);
  1829. END;
  1830. INC(sadr,4); INC(dadr,4); DEC(len);
  1831. END
  1832. END BGRA8888OverBGRA8888;
  1833. (* BGRA8888 *)
  1834. PROCEDURE BGRA8888OverAny (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1835. VAR spix, dpix: Pixel;
  1836. BEGIN
  1837. WHILE len > 0 DO
  1838. SYSTEM.MOVE(sadr, ADDRESSOF(spix),4);
  1839. (*
  1840. SYSTEM.GET (sadr, spix);
  1841. *)
  1842. IF spix[a] = 0FFX THEN
  1843. mode.dst.pack(mode.dst, dadr, dbit, spix)
  1844. ELSIF spix[a] # 0X THEN
  1845. mode.dst.unpack(mode.dst, dadr, dbit, dpix);
  1846. Blend(mode.op, spix, dpix);
  1847. mode.dst.pack(mode.dst, dadr, dbit, dpix)
  1848. END;
  1849. INC(sadr, SIZEOF (Pixel)); DEC(len);
  1850. dbit := dbit + mode.dst.bpp; INC(dadr, dbit DIV 8); dbit := dbit MOD 8
  1851. END
  1852. END BGRA8888OverAny;
  1853. PROCEDURE BGRA8888OverA8 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1854. VAR in, out: CHAR;
  1855. BEGIN
  1856. WHILE len > 0 DO
  1857. SYSTEM.GET(sadr + a, in);
  1858. IF in = 0FFX THEN
  1859. SYSTEM.PUT(dadr, 0FFX)
  1860. ELSIF in # 0X THEN
  1861. SYSTEM.GET(dadr, out);
  1862. SYSTEM.PUT(dadr, CHR(ORD(in) + ORD(out) * (255 - ORD(in)) DIV 255))
  1863. END;
  1864. INC(sadr, 4); INC(dadr); DEC(len)
  1865. END
  1866. END BGRA8888OverA8;
  1867. PROCEDURE BGRA8888OverAny16 (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1868. VAR src, dst: Pixel; fd, t: LONGINT;
  1869. BEGIN
  1870. WHILE len > 0 DO
  1871. SYSTEM.MOVE(sadr,ADDRESSOF(src),4);
  1872. (* SYSTEM.GET (sadr, src);*)
  1873. IF src[a] = 0FFX THEN
  1874. mode.dst.pack(mode.dst, dadr, dbit, src)
  1875. ELSIF src[a] # 0X THEN
  1876. mode.dst.unpack(mode.dst, dadr, 0, dst);
  1877. fd := 255-ORD(src[a]);
  1878. t := (256 * ORD(src[0]) + fd * ORD(dst[0])) DIV 256;
  1879. IF t < 255 THEN dst[0] := CHR(t) ELSE dst[0] := CHR(255) END;
  1880. t := (256 * ORD(src[1]) + fd * ORD(dst[1])) DIV 256;
  1881. IF t < 255 THEN dst[1] := CHR(t) ELSE dst[1] := CHR(255) END;
  1882. t := (256 * ORD(src[2]) + fd * ORD(dst[2])) DIV 256;
  1883. IF t < 255 THEN dst[2] := CHR(t) ELSE dst[2] := CHR(255) END;
  1884. t := (256 * ORD(src[3]) + fd * ORD(dst[3])) DIV 256;
  1885. IF t < 255 THEN dst[3] := CHR(t) ELSE dst[3] := CHR(255) END;
  1886. mode.dst.pack(mode.dst, dadr, 0, dst);
  1887. END;
  1888. INC(dadr, 2); INC(sadr, SIZEOF (Pixel)); DEC(len)
  1889. END
  1890. END BGRA8888OverAny16;
  1891. PROCEDURE BGRA8888Over565* (VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1892. VAR src, dst: Pixel; fd, t: LONGINT;
  1893. BEGIN
  1894. WHILE len > 0 DO
  1895. SYSTEM.MOVE(sadr,ADDRESSOF(src),4);
  1896. (*
  1897. SYSTEM.GET (sadr, src);
  1898. *)
  1899. IF src[a] = 0FFX THEN
  1900. SYSTEM.PUT16(dadr, ASH(ORD(src[b]), -3) + ASH(ASH(ORD(src[g]), -2), 5) + ASH(ASH(ORD(src[r]), -3), 11));
  1901. ELSIF src[a] # 0X THEN
  1902. t := SYSTEM.GET16(dadr);
  1903. dst[b] := CHR((t MOD 32) * 8); dst[g] := CHR((t DIV 32 MOD 64) * 4); dst[r] := CHR((t DIV 2048 MOD 32) * 8);
  1904. fd := 256-ORD(src[a]);
  1905. t := (256 * ORD(src[0]) + fd * ORD(dst[0])) DIV 256;
  1906. IF t < 255 THEN dst[0] := CHR(t) ELSE dst[0] := CHR(255) END;
  1907. t := (256 * ORD(src[1]) + fd * ORD(dst[1])) DIV 256;
  1908. IF t < 255 THEN dst[1] := CHR(t) ELSE dst[1] := CHR(255) END;
  1909. t := (256 * ORD(src[2]) + fd * ORD(dst[2])) DIV 256;
  1910. IF t < 255 THEN dst[2] := CHR(t) ELSE dst[2] := CHR(255) END;
  1911. t := (256 * ORD(src[3]) + fd * ORD(dst[3])) DIV 256;
  1912. IF t < 255 THEN dst[3] := CHR(t) ELSE dst[3] := CHR(255) END;
  1913. SYSTEM.PUT16(dadr, ASH(ORD(dst[b]), -3) + ASH(ASH(ORD(dst[g]), -2), 5) + ASH(ASH(ORD(dst[r]), -3), 11));
  1914. END;
  1915. INC(dadr, 2); INC(sadr, 4); DEC(len)
  1916. END
  1917. END BGRA8888Over565;
  1918. PROCEDURE MMXBGRA8888Over565(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  1919. CODE {SYSTEM.i386, SYSTEM.MMX}
  1920. PUSHFD
  1921. PUSH ECX
  1922. ; CLI
  1923. MOV ESI, [EBP+sadr]
  1924. MOV EDI, [EBP+dadr]
  1925. PXOR MMX0, MMX0
  1926. PXOR MMX1, MMX1
  1927. MOV EAX, 0FFFFFFFFH
  1928. MOVD MMX7, EAX
  1929. PUNPCKLBW MMX7, MMX0 ; 00FF00FF00FF00FF
  1930. MOV ECX, [EBP+len]
  1931. loop:
  1932. CMP ECX, 0
  1933. JE end
  1934. MOV EAX, [ESI]
  1935. XOR EBX, EBX
  1936. MOV BX, [EDI]
  1937. ; 255 - alpha
  1938. MOV EDX, EAX
  1939. SHR EDX, 24
  1940. CMP EDX, 0
  1941. JE empty
  1942. CMP EDX, 255
  1943. JE full
  1944. alpha:
  1945. NEG EDX
  1946. ADD EDX, 255
  1947. MOVD MMX6, EDX
  1948. PUNPCKLWD MMX6, MMX6
  1949. PUNPCKLDQ MMX6, MMX6
  1950. MOVD MMX1, EAX
  1951. ; unpack dst
  1952. MOV EDX, EBX ; b
  1953. SHL EDX, 3
  1954. AND EDX, 0F8H
  1955. MOV EAX, EDX
  1956. MOV EDX, EBX ; g
  1957. SHL EDX, 5
  1958. AND EDX, 0FC00H
  1959. OR EAX, EDX
  1960. MOV EDX, EBX ; r
  1961. SHL EDX, 8
  1962. AND EDX, 0F80000H
  1963. OR EAX, EDX
  1964. MOVD MMX2, EAX
  1965. PUNPCKLBW MMX1, MMX0 ; 0000ARGB --> 0A0R0G0B
  1966. PMULLW MMX1, MMX7
  1967. PUNPCKLBW MMX2, MMX0 ; 0000ARGB --> 0A0R0G0B
  1968. PMULLW MMX2, MMX6
  1969. PADDUSW MMX1, MMX2
  1970. ; PSRLW MMX1, 8 ; normalize
  1971. DB 0FH, 71H, 0D1H, 08H
  1972. PACKUSWB MMX1, MMX0
  1973. ; HUGA BIMBO Muell
  1974. MOVD EAX, MMX1
  1975. full:
  1976. ; XOR EDX, EDX
  1977. ; SHR EAX, 3
  1978. ; MOV EDX, EAX
  1979. ; AND EDX, 1FH
  1980. ; SHR EAX, 2
  1981. ; AND EAX, 0FFFFFFE0H
  1982. ; OR EDX, EAX
  1983. ; AND EDX, 7FFH
  1984. ;
  1985. ; SHR EAX, 3
  1986. ; AND EAX,
  1987. ;
  1988. ;
  1989. ; SHR AL, 3
  1990. ; SHR AH, 2
  1991. ; MOV EDX, EAX
  1992. ; SHR EAX, 3
  1993. ; AND EAX, 01F0000H
  1994. ; OR EDX, EAX
  1995. ; AND EDX, 01F3F1FH
  1996. MOV EBX, EAX
  1997. AND EBX, 0FFH
  1998. SHR EBX, 3
  1999. MOV EDX, EBX
  2000. MOV EBX, EAX
  2001. SHR EBX, 8
  2002. AND EBX, 0FFH
  2003. SHR EBX, 2
  2004. SHL EBX, 5
  2005. OR EDX, EBX
  2006. MOV EBX, EAX
  2007. SHR EBX, 16
  2008. AND EBX, 0FFH
  2009. SHR EBX, 3
  2010. SHL EBX, 11
  2011. OR EDX, EBX
  2012. MOV [EDI], DX
  2013. empty:
  2014. ADD ESI, 4;
  2015. ADD EDI, 2;
  2016. DEC ECX
  2017. JMP loop
  2018. end:
  2019. EMMS ; declare FPU registers free
  2020. POP ECX
  2021. POPFD
  2022. END MMXBGRA8888Over565;
  2023. PROCEDURE SSE2BGRA8888Over565(VAR mode: Mode0; sadr: ADDRESS; sbit: LONGINT; dadr: ADDRESS; dbit, len: LONGINT);
  2024. CODE {SYSTEM.i386, SYSTEM.MMX, SYSTEM.SSE, SYSTEM.SSE2}
  2025. PUSHFD
  2026. PUSH ECX
  2027. PUSH EBX
  2028. ; CLI
  2029. PXOR MMX0, MMX0
  2030. PXOR MMX1, MMX1
  2031. PXOR MMX2, MMX2
  2032. PXOR MMX3, MMX3
  2033. PXOR MMX4, MMX4
  2034. PXOR MMX5, MMX5
  2035. PXOR MMX6, MMX6
  2036. PXOR MMX7, MMX7
  2037. PXOR XMM3, XMM3
  2038. PXOR XMM4, XMM4
  2039. MOV ESI, [EBP+sadr]
  2040. MOV EDI, [EBP+dadr]
  2041. MOV ECX, [EBP+len]
  2042. ; create masks
  2043. ; src only FF which is rotated -> MMX3
  2044. MOV EAX, 0000000FFH
  2045. MOVD MMX3, EAX
  2046. ; dest red -> MMX4
  2047. MOV EAX, 0F800F800H
  2048. MOVD MMX4, EAX
  2049. ; dest green -> MMX5
  2050. MOV EAX, 07E007E0H
  2051. MOVD MMX5, EAX
  2052. ; dest blue -> MMX6 ; moved as MMX6 is used in singlepixel
  2053. ; MOV EAX, 001F001FH
  2054. ; MOVD MMX6, EAX
  2055. ; BEGIN
  2056. ; WHILE len > 0 DO
  2057. loop:
  2058. CMP ECX,0
  2059. JE end ; jump to end if ECX = 0
  2060. ; if len < 8 then do one pixel at the time
  2061. CMP ECX, 8
  2062. JL singlepixel
  2063. ; else
  2064. ; take 8 at the time
  2065. MOV EBX, ESI
  2066. AND EBX, 0FH
  2067. CMP EBX, 0
  2068. JNE singlepixel
  2069. alleightpixels:
  2070. ; dest blue -> MMX6
  2071. MOV EAX, 001F001FH
  2072. MOVD MMX6, EAX
  2073. ; src := SYSTEM.VAL(Pixel, SYSTEM.GET32(sadr));
  2074. ; Load data into memory
  2075. ;MOV XMM4, 0FF000000FF000000FF000000FF000000H
  2076. MOVDQA XMM2, [ESI] ;src 5-8
  2077. MOVQ2DQ XMM4, MMX3 ; 000000000000000000000000000000FFH
  2078. MOVDQA XMM1, [ESI+16] ;src 1-4
  2079. PREFETCHNTA [ESI+32] ; prepare src 9-15
  2080. ; get alphas
  2081. MOVDQU XMM6, XMM2
  2082. PSHUFD XMM4, XMM4, 0
  2083. MOVDQU XMM5, XMM1
  2084. PSLLD XMM4, 24
  2085. PAND XMM6, XMM4 ; alpha 5-8 in XMM6
  2086. PAND XMM5, XMM4 ; alpha 1-4 in XMM5
  2087. PSRLD XMM5, 24
  2088. PSHUFHW XMM5, XMM5, 85H
  2089. PSRLD XMM6, 24
  2090. ; put both alphas into 1 register
  2091. PSHUFHW XMM6, XMM6, 85H
  2092. PSHUFLW XMM5, XMM5, 85H
  2093. PSHUFLW XMM6, XMM6, 58H
  2094. PSHUFD XMM5, XMM5, 0D0H ; 0102030400000000
  2095. PSHUFD XMM6, XMM6, 5CH ; 0000000005060708
  2096. PXOR XMM0,XMM0
  2097. POR XMM5, XMM6 ; XMM5 = alphas 0102030405060708
  2098. PCMPEQD XMM0, XMM5
  2099. PMOVMSKB EAX, XMM0
  2100. CMP EAX, 0FFFFH ; all alphas = zero; TEST not possible, because only 8 bits compared
  2101. JE endloop
  2102. ; mask out alpha = zero
  2103. ; fd := 255-ORD(src[a]); fd = XMM4
  2104. ; MOV XMM4, 00FF00FF00FF00FF00FF00FF00FF00FFH
  2105. PXOR XMM4, XMM4
  2106. MOV EAX, 00FFH
  2107. PINSRW XMM4, EAX ,0
  2108. PSHUFLW XMM4, XMM4, 0
  2109. PSHUFD XMM4, XMM4, 0
  2110. PSUBW XMM4, XMM5
  2111. MOV EAX,1H
  2112. PINSRW XMM3, EAX ,0
  2113. PSHUFLW XMM3, XMM3, 0
  2114. PSHUFD XMM3, XMM3, 0
  2115. PADDUSW XMM4, XMM3
  2116. ; new red
  2117. ; calculate red 2
  2118. ; get source
  2119. ; sred14 = src14 && (srcMask <<16)
  2120. ; srcMask << 16
  2121. MOVQ2DQ XMM3, MMX3
  2122. PSHUFD XMM3, XMM3, 0
  2123. MOVDQU XMM5, XMM1
  2124. MOVDQU XMM6, XMM2
  2125. PSLLD XMM3, 16
  2126. ; sred14 = src14 && (srcMask << 24)
  2127. ; src14 must be copied because it mustn't be changed
  2128. PAND XMM5, XMM3 ; sred14
  2129. PSRLD XMM5, 16
  2130. ; sred14s = shuffled sred14
  2131. PSHUFHW XMM5, XMM5,85H
  2132. PAND XMM6, XMM3 ; sred58
  2133. PSRLD XMM6, 16
  2134. PSHUFLW XMM5, XMM5,85H
  2135. PSHUFHW XMM6, XMM6,85H
  2136. PSHUFD XMM5, XMM5,0D0H ; sred14s
  2137. PSHUFLW XMM6, XMM6,58H
  2138. PSHUFD XMM6, XMM6,5CH ; sred58s
  2139. POR XMM5, XMM6 ; sred18
  2140. ; sred18255 = sred18 * 256- sred18
  2141. MOVDQU XMM7, XMM5
  2142. PSLLW XMM5, 8
  2143. PSUBUSW XMM5, XMM7 ; sred18255
  2144. ; src is now ready
  2145. ; destination
  2146. ; dest18 must be copied because it mustn't be changed
  2147. ; Load data into memory
  2148. MOVDQU XMM3, [EDI] ;dest 1-8
  2149. MOVQ2DQ XMM6, MMX4
  2150. PSHUFD XMM6, XMM6, 0
  2151. MOVDQU XMM7, XMM3
  2152. PAND XMM7, XMM6 ; dred18
  2153. PSRLW XMM7, 8
  2154. ; dred18alpha = dred18 * negalpha
  2155. PMULLW XMM7, XMM4 ; dred18alpha
  2156. ; dest is prepared
  2157. ; combining dest and src
  2158. ; dred18big = sred18255 + dred18alpha
  2159. PADDUSW XMM7, XMM5 ; dred18big
  2160. ; dred18f = dred18big && destMaskred128 because >> 11 and << 11 is && mask
  2161. PAND XMM7, XMM6 ; dred18f
  2162. ; dest18nr0 = dest18 && (~destMaskred128)
  2163. PANDN XMM6, XMM3 ; dest18nr0
  2164. ; dest18nrf = dest18nr0 || dred18f
  2165. POR XMM6, XMM7
  2166. MOVDQU XMM3, XMM6
  2167. ; red is calculated
  2168. ; calculate green:
  2169. ; get source
  2170. ; sgreen14 = src14 && (srcMask <<8)
  2171. ; srcMask << 8
  2172. MOVQ2DQ XMM7, MMX3
  2173. PSHUFD XMM7, XMM7, 0
  2174. MOVDQU XMM5, XMM1
  2175. PSLLD XMM7, 8
  2176. PAND XMM5, XMM7 ; sgreen14
  2177. PSRLD XMM5, 8
  2178. ; sgreen14s = shuffled sgreen14
  2179. PSHUFHW XMM5, XMM5,85H
  2180. MOVDQU XMM6, XMM2
  2181. PSHUFLW XMM5, XMM5,85H
  2182. PAND XMM6, XMM7 ; sgreen58
  2183. PSRLD XMM6, 8
  2184. PSHUFD XMM5, XMM5,0D0H ; sgreen14s
  2185. ; sgreen58 = src58&& (srcMask << 8)
  2186. ; src58 must be copied because it mustn't be changed
  2187. ; sgreen58s = shuffled sgreen58
  2188. PSHUFHW XMM6, XMM6,85H
  2189. PSHUFLW XMM6, XMM6,58H
  2190. PSHUFD XMM6, XMM6,5CH ; sgreen58s
  2191. ; sgreen18 = sgreen14s || sgreen58s
  2192. POR XMM5, XMM6 ; sgreen18
  2193. ; sgreen18255 = sgreen18 * 256- sgreen18
  2194. MOVDQU XMM7, XMM5
  2195. MOVQ2DQ XMM6, MMX5
  2196. PSLLW XMM5, 8
  2197. PSUBUSW XMM5, XMM7 ; sgreen18255
  2198. PSHUFD XMM6, XMM6, 0
  2199. MOVDQU XMM7, XMM3
  2200. PAND XMM7, XMM6 ; dgreen18
  2201. PSRLW XMM7,3
  2202. ; dgreen18alpha = dgreen18 * negalpha
  2203. PMULLW XMM7, XMM4 ; dgreen18alpha
  2204. ; dest is prepared
  2205. ; combining dest and src
  2206. ; dgreen18big = sgreen18255 + dgreen18alpha
  2207. PADDUSW XMM7, XMM5 ; dgreen18big
  2208. PANDN XMM6, XMM3 ; dest18ng0
  2209. ; dgreen18f = (dgreen18big >> 11) <<5
  2210. PSRLW XMM7, 10 ; dgreen18f
  2211. PSLLW XMM7, 5
  2212. ; dest18ng0 = dest18 && (~destMaskgreen128)
  2213. ; dest18ngf = dest18ng0 || dred18f
  2214. POR XMM6, XMM7
  2215. MOVDQU XMM3, XMM6
  2216. ; green is calculated
  2217. ; calculate blue
  2218. ; get source
  2219. ; sblue14 = src14 && (srcMask)
  2220. ; srcMask
  2221. MOVQ2DQ XMM7, MMX3
  2222. MOVDQU XMM5, XMM1
  2223. PSHUFD XMM7, XMM7, 0
  2224. MOVDQU XMM6, XMM2
  2225. ; sblue14 = src14 && (srcMask)
  2226. ; src14 must be copied because it mustn't be changed
  2227. PAND XMM5, XMM7 ; sblue14
  2228. ; sblue14s = shuffled sblue14
  2229. PSHUFHW XMM5, XMM5,85H
  2230. PAND XMM6, XMM7 ; sblue58
  2231. PSHUFHW XMM6, XMM6,85H
  2232. PSHUFLW XMM5, XMM5,85H
  2233. PSHUFLW XMM6, XMM6,58H
  2234. PSHUFD XMM5, XMM5,0D0H ; sblue14s
  2235. PSHUFD XMM6, XMM6,5CH ; sblue58s
  2236. POR XMM5, XMM6 ; sblue18
  2237. ; sblue18255 = sblue18 * 256- sblue18
  2238. MOVDQU XMM7, XMM5
  2239. PSLLW XMM5, 8
  2240. PSUBUSW XMM5, XMM7 ; sblue18255
  2241. MOVQ2DQ XMM6, MMX6
  2242. PSHUFD XMM6, XMM6, 0
  2243. MOVDQU XMM7, XMM3
  2244. PAND XMM7, XMM6 ; dblue18
  2245. PSLLW XMM7, 3
  2246. PMULLW XMM7, XMM4 ; dblue18alpha
  2247. ; dest is prepared
  2248. ; combining dest and src
  2249. ; dblue18big = sblue18255 + dblue18alpha
  2250. PADDUSW XMM7, XMM5 ; dblue18big
  2251. ; dblue18f = (dblue18big >> 11)
  2252. PANDN XMM6, XMM3 ; dest18nr0
  2253. PSRLW XMM7, 11 ; dblue18f
  2254. ; dest18nr0 = dest18 && (~destMaskblue128)
  2255. ; dest18nbf = dest18nb0 || dblue18f
  2256. POR XMM6, XMM7
  2257. MOVDQU XMM3, XMM6
  2258. ; blue is calculated
  2259. ; now dest is calculated, store it
  2260. ; get 0 stuff
  2261. MOVDQU XMM5, [EDI]
  2262. PAND XMM5,XMM0
  2263. PANDN XMM0, XMM3
  2264. POR XMM0, XMM5
  2265. MOVDQU [EDI],XMM0
  2266. PREFETCHNTA [EDI+16] ; prepare dest 9-15
  2267. endloop:
  2268. ADD ESI, 32 ; num of bytes
  2269. ADD EDI, 16
  2270. SUB ECX, 8
  2271. JMP loop
  2272. singlepixel: ; original code from MMXBGRA8888Over565, adjusted to fit this procedure
  2273. MOV EAX, 0FFFFFFFFH
  2274. MOVD MMX7, EAX
  2275. PUNPCKLBW MMX7, MMX0 ; 00FF00FF00FF00FF
  2276. MOV EAX,[ESI]
  2277. XOR EBX, EBX
  2278. MOV BX, [EDI]
  2279. ; 255 - alpha
  2280. MOV EDX, EAX
  2281. SHR EDX, 24
  2282. CMP EDX, 0
  2283. JE empty
  2284. CMP EDX, 255
  2285. JE full
  2286. alpha:
  2287. NEG EDX
  2288. ADD EDX, 255
  2289. MOVD MMX6, EDX
  2290. PUNPCKLWD MMX6, MMX6
  2291. PUNPCKLDQ MMX6, MMX6
  2292. MOVD MMX1, EAX
  2293. ; unpack dst
  2294. MOV EDX, EBX ; b
  2295. SHL EDX, 3
  2296. AND EDX, 0F8H
  2297. MOV EAX, EDX
  2298. MOV EDX, EBX ; g
  2299. SHL EDX, 5
  2300. AND EDX, 0FC00H
  2301. OR EAX, EDX
  2302. MOV EDX, EBX ; r
  2303. SHL EDX, 8
  2304. AND EDX, 0F80000H
  2305. OR EAX, EDX
  2306. MOVD MMX2, EAX
  2307. PUNPCKLBW MMX1, MMX0 ; 0000ARGB --> 0A0R0G0B
  2308. PMULLW MMX1, MMX7
  2309. PUNPCKLBW MMX2, MMX0 ; 0000ARGB --> 0A0R0G0B
  2310. PMULLW MMX2, MMX6
  2311. PADDUSW MMX1, MMX2
  2312. ; PSRLW MMX1, 8 ; normalize
  2313. DB 0FH, 71H, 0D1H, 08H
  2314. PACKUSWB MMX1, MMX0
  2315. ; HUGA BIMBO Muell
  2316. MOVD EAX, MMX1
  2317. full:
  2318. MOV EBX, EAX
  2319. AND EBX, 0FFH
  2320. SHR EBX, 3
  2321. MOV EDX, EBX
  2322. MOV EBX, EAX
  2323. SHR EBX, 8
  2324. AND EBX, 0FFH
  2325. SHR EBX, 2
  2326. SHL EBX, 5
  2327. OR EDX, EBX
  2328. MOV EBX, EAX
  2329. SHR EBX, 16
  2330. AND EBX, 0FFH
  2331. SHR EBX, 3
  2332. SHL EBX, 11
  2333. OR EDX, EBX
  2334. MOV [EDI], DX
  2335. empty:
  2336. ADD ESI, 4;
  2337. ADD EDI, 2;
  2338. DEC ECX
  2339. JMP loop
  2340. end:
  2341. EMMS ; declare FPU registers free
  2342. POP EBX
  2343. POP ECX
  2344. POPFD
  2345. END SSE2BGRA8888Over565;
  2346. (** find (optimized) pixel transfer procedure for transfer mode and given source and destination formats **)
  2347. PROCEDURE Bind* (VAR mode: Mode; VAR src, dst: Format);
  2348. VAR op: LONGINT; val,i: LONGINT;
  2349. BEGIN
  2350. IF Same(src, mode.src) & Same(dst, mode.dst) THEN
  2351. ASSERT(mode.transfer # NIL, 120);
  2352. RETURN (* assume transfer procedure is still valid *)
  2353. END;
  2354. mode.src := src; mode.dst := dst; mode.buf := mode.col;
  2355. IF (src.pal # NIL) & ((mode.map = NIL) OR (LEN(mode.map^) # src.pal.used) ) THEN
  2356. NEW(mode.map, src.pal.used)
  2357. END;
  2358. (* try to convert complex composite operations into simpler ones *)
  2359. IF alpha IN src.components * dst.components THEN (* source and destination contain alpha information *)
  2360. op := mode.op
  2361. ELSIF alpha IN src.components THEN (* only source contains alpha *)
  2362. CASE mode.op OF
  2363. | dstOverSrc: op := dstCopy
  2364. | srcInDst: op := srcCopy
  2365. | srcWithoutDst: op := clear
  2366. | srcAtopDst: op := srcOverDst
  2367. | dstAtopSrc: op := dstInSrc
  2368. | srcXorDst: op := dstWithoutSrc
  2369. ELSE op := mode.op
  2370. END
  2371. ELSIF alpha IN dst.components THEN (* only destination contains alpha *)
  2372. CASE mode.op OF
  2373. | srcOverDst: op := srcCopy
  2374. | dstInSrc: op := dstCopy
  2375. | dstWithoutSrc: op := clear
  2376. | srcAtopDst: op := srcInDst
  2377. | dstAtopSrc: op := dstOverSrc
  2378. | srcXorDst: op := srcWithoutDst
  2379. ELSE op := mode.op
  2380. END
  2381. ELSE (* no alpha in either source or destination *)
  2382. CASE mode.op OF
  2383. | srcOverDst, srcInDst, srcAtopDst: op := srcCopy
  2384. | dstOverSrc, dstInSrc, dstAtopSrc: op := dstCopy
  2385. | srcWithoutDst, dstWithoutSrc, srcXorDst: op := clear
  2386. ELSE op := mode.op
  2387. END
  2388. END;
  2389. IF op = InvDst THEN
  2390. mode.transfer:=InvAny;
  2391. ELSIF op = InvOverDst THEN
  2392. mode.transfer:=InvOverAny;
  2393. ELSIF op = clear THEN
  2394. CASE dst.code OF
  2395. | a1: mode.transfer := Clear1
  2396. | a8, bgr555, bgr565, bgr466, bgr888(*, bgra8888*): mode.transfer := ClearBytes
  2397. | bgra8888: mode.transfer:=Clear32;
  2398. | p8:
  2399. mode.buf[0] := CHR(CLUTs.Match(dst.pal.clut, 0));
  2400. IF mode.buf[0] = 0X THEN mode.transfer := ClearBytes
  2401. ELSE mode.transfer := ConstCopy8
  2402. END
  2403. | d8:
  2404. mode.buf[0] := CHR(ColorToIndex(0));
  2405. IF mode.buf[0] = 0X THEN mode.transfer := ClearBytes
  2406. ELSE mode.transfer := ConstCopy8
  2407. END
  2408. | p16:
  2409. val:=CLUTs.Match(dst.pal.clut, 0);
  2410. IF dst.pal.used>256 THEN val:=val*dst.pal.used DIV 256 END;
  2411. SYSTEM.PUT16(ADDRESSOF(mode.buf[0]),SHORT(val)); (*PH090122*)
  2412. (*mode.buf[0] := CHR(PaletteIndex(dst.pal, 0, 0, 0));*)
  2413. IF val = 0 THEN mode.transfer := ClearBytes
  2414. ELSE mode.transfer := ConstCopy16
  2415. END
  2416. ELSE mode.transfer := ClearAny
  2417. END
  2418. ELSIF op = srcCopy THEN
  2419. CASE dst.code OF
  2420. | a1:
  2421. CASE src.code OF
  2422. | a1: mode.transfer := Copy1
  2423. | a8: mode.transfer := A8CopyA1
  2424. | bgra8888: mode.transfer := BGRA8888CopyA1
  2425. ELSE
  2426. IF alpha IN src.components THEN mode.transfer := AnyCopyA1
  2427. ELSE mode.transfer := Set1
  2428. END
  2429. END
  2430. | a8:
  2431. CASE src.code OF
  2432. | a1: mode.transfer := A1CopyA8
  2433. | a8: mode.transfer := Copy8
  2434. | bgra8888: mode.transfer := BGRA8888CopyA8
  2435. ELSE
  2436. IF alpha IN src.components THEN mode.transfer := AnyCopyA8
  2437. ELSE mode.buf[0] := 0FFX; mode.transfer := ConstCopy8
  2438. END
  2439. END
  2440. | p8:
  2441. CASE src.code OF
  2442. | a1, a8:
  2443. mode.buf[0] := CHR(CLUTs.Match(dst.pal.clut, ORD(mode.col[b]) + ASH(ORD(mode.col[g]), 8) +
  2444. ASH(ORD(mode.col[r]), 16)));
  2445. mode.transfer := ConstCopy8
  2446. | p8:
  2447. IF src.pal = dst.pal THEN mode.transfer := Copy8
  2448. ELSE
  2449. FOR i := 0 TO src.pal.used-1 DO
  2450. mode.map[i] := SHORT(CLUTs.Match(dst.pal.clut, ORD(src.pal.col[i, b]) + ASH(ORD(src.pal.col[i, g]), 8) +
  2451. ASH(ORD(src.pal.col[i, r]), 16)))
  2452. END;
  2453. mode.transfer := I8CopyI8
  2454. END
  2455. | d8:
  2456. FOR i := 0 TO 255 DO
  2457. mode.map[i] := SHORT(CLUTs.Match(dst.pal.clut, IndexToColor(i) MOD 1000000H))
  2458. END;
  2459. mode.transfer := I8CopyI8
  2460. | bgr555, bgr565, bgr466,p16: mode.transfer := Any16CopyP8
  2461. | bgr888: mode.transfer := BGR888CopyP8
  2462. | bgra8888: mode.transfer := BGRA8888CopyP8
  2463. ELSE mode.transfer := AnyCopyP8
  2464. END
  2465. | d8:
  2466. CASE src.code OF
  2467. | a1, a8:
  2468. mode.buf[0] := CHR(ColorToIndex(
  2469. ORD(mode.col[b]) + ASH(ORD(mode.col[g]), 8) + ASH(ORD(mode.col[r]), 16)));
  2470. mode.transfer := ConstCopy8
  2471. | p8:
  2472. FOR i := 0 TO src.pal.used-1 DO
  2473. mode.map[i] := SHORT(ColorToIndex(
  2474. ORD(src.pal.col[i, b]) + ASH(ORD(src.pal.col[i, g]), 8) + ASH(ORD(src.pal.col[i, r]), 16)))
  2475. END;
  2476. mode.transfer := I8CopyI8
  2477. | d8: mode.transfer := Copy8
  2478. | bgr555, bgr565, bgr466, p16: mode.transfer := Any16CopyD8
  2479. | bgr888: mode.transfer := BGR888CopyD8
  2480. | bgra8888: mode.transfer := BGRA8888CopyD8
  2481. ELSE mode.transfer := AnyCopyD8
  2482. END
  2483. | p16:
  2484. CASE src.code OF
  2485. | a1, a8: dst.pack(dst, ADDRESSOF(mode.buf[0]), 0, mode.col); mode.transfer := ConstCopy16
  2486. | p8: mode.transfer := P8CopyAny16
  2487. | d8: mode.transfer := D8CopyAny16
  2488. | p16:
  2489. IF src.pal = dst.pal THEN mode.transfer := Copy16
  2490. ELSE
  2491. FOR i := 0 TO src.pal.used-1 DO
  2492. val:=CLUTs.Match(dst.pal.clut, ORD(src.pal.col[i, b]) + ASH(ORD(src.pal.col[i, g]), 8) +
  2493. ASH(ORD(src.pal.col[i, r]), 16));
  2494. IF dst.pal.used>256 THEN val := val * dst.pal.used DIV 256 END;
  2495. mode.map[i] := SHORT(val)
  2496. END;
  2497. mode.transfer := I16CopyI16
  2498. END
  2499. | bgr555, bgr565, bgr466: mode.transfer := Any16CopyAny16
  2500. | bgr888: mode.transfer := BGR888CopyAny16
  2501. | bgra8888: mode.transfer := BGRA8888CopyAny16
  2502. ELSE mode.transfer := AnyCopyAny16
  2503. END;
  2504. | bgr555, bgr565, bgr466:
  2505. CASE src.code OF
  2506. | a1, a8: dst.pack(dst, ADDRESSOF(mode.buf[0]), 0, mode.col); mode.transfer := ConstCopy16
  2507. | p8: mode.transfer := P8CopyAny16
  2508. | d8: mode.transfer := D8CopyAny16
  2509. | bgr555, bgr565, bgr466,p16:
  2510. IF src.code = dst.code THEN mode.transfer := Copy16
  2511. ELSE mode.transfer := Any16CopyAny16
  2512. END
  2513. | bgr888: mode.transfer := BGR888CopyAny16
  2514. | bgra8888: mode.transfer := BGRA8888CopyAny16
  2515. ELSE mode.transfer := AnyCopyAny16
  2516. END;
  2517. | bgr888:
  2518. CASE src.code OF
  2519. | a1, a8: mode.buf := mode.col; mode.transfer := ConstCopy24
  2520. | p8: mode.transfer := P8CopyBGR888
  2521. | d8: mode.transfer := D8CopyBGR888
  2522. | p16: mode.transfer := P16CopyBGR888 (*PH090122*)
  2523. | bgr555, bgr565, bgr466: mode.transfer := Any16CopyBGR888
  2524. | bgr888: mode.transfer := Copy24
  2525. | bgra8888:
  2526. (* IF SSE2enabled THEN mode.transfer := SSE2BGRA8888CopyBGR888
  2527. ELSE *) mode.transfer := BGRA8888CopyBGR888
  2528. (* END*);
  2529. ELSE mode.transfer := AnyCopyBGR888
  2530. END
  2531. | bgra8888:
  2532. CASE src.code OF
  2533. | a1: mode.transfer := A1CopyBGRA8888
  2534. | a8: mode.transfer := A8CopyBGRA8888
  2535. | p8: mode.transfer := P8CopyBGRA8888
  2536. | d8: mode.transfer := D8CopyBGRA8888
  2537. | p16: mode.transfer := P16CopyBGRA8888 (*PH090122*)
  2538. | bgr555, bgr565, bgr466: mode.transfer := Any16CopyBGRA8888
  2539. | bgr888: IF SSE2enabled THEN mode.transfer := SSE2BGR888CopyBGRA8888
  2540. ELSE mode.transfer :=BGR888CopyBGRA8888;
  2541. END;
  2542. | bgra8888: mode.transfer := Copy32
  2543. ELSE mode.transfer := AnyCopyBGRA8888
  2544. END
  2545. ELSE
  2546. CASE src.code OF
  2547. | a1: mode.transfer := A1CopyAny
  2548. | p8: mode.transfer := P8CopyAny
  2549. | d8: mode.transfer := D8CopyAny
  2550. | bgr555, bgr565, bgr466,p16: mode.transfer := Any16CopyAny
  2551. | bgr888: mode.transfer := BGR888CopyAny
  2552. | bgra8888: mode.transfer := BGRA8888CopyAny
  2553. ELSE
  2554. IF (src.bpp MOD 8 = 0) & (dst.bpp MOD 8 = 0) THEN mode.transfer := AnyBytesCopyAnyBytes
  2555. ELSE mode.transfer := AnyCopyAny
  2556. END
  2557. END
  2558. END
  2559. ELSIF op = dstOverSrc THEN
  2560. mode.transfer := EmptyTransfer
  2561. ELSIF op = srcOverDst THEN
  2562. CASE dst.code OF
  2563. | a1:
  2564. CASE src.code OF
  2565. | a1: mode.transfer := A1OverA1
  2566. | a8: mode.transfer := A8OverA1
  2567. | bgra8888: mode.transfer := BGRA8888OverA1
  2568. ELSE mode.transfer := AnyOverA1
  2569. END
  2570. | a8:
  2571. CASE src.code OF
  2572. | a1: mode.buf[0] := 0FFX; mode.transfer := A1OverConst8
  2573. | a8: mode.transfer := A8OverA8
  2574. | bgra8888: mode.transfer := BGRA8888OverA8
  2575. ELSE mode.transfer := AnyOverA8
  2576. END
  2577. | bgra8888:
  2578. CASE src.code OF
  2579. | a1: mode.buf := mode.col; mode.transfer := A1OverConst32
  2580. | a8: mode.buf := mode.col; mode.transfer := A8OverAny;
  2581. IF mode.op = srcOverDst THEN mode.transfer := A8OverBGRA8888 END;
  2582. | bgra8888: mode.transfer := BGRA8888OverBGRA8888
  2583. ELSE mode.transfer := BGRA8888OverAny; (* ? *)
  2584. END
  2585. ELSE
  2586. CASE src.code OF
  2587. | a1:
  2588. CASE dst.code OF
  2589. | p8:
  2590. mode.buf[0] := CHR(CLUTs.Match(dst.pal.clut, ORD(mode.col[b]) + ASH(ORD(mode.col[g]), 8) +
  2591. ASH(ORD(mode.col[r]), 16)));
  2592. mode.transfer := A1OverConst8
  2593. | d8:
  2594. mode.buf[0] := CHR(ColorToIndex(ORD(mode.col[b]) + ASH(ORD(mode.col[g]), 8) +
  2595. ASH(ORD(mode.col[r]), 16)));
  2596. mode.transfer := A1OverConst8
  2597. | p16: (* this is probably not correct ... *)
  2598. mode.buf[0] := CHR(PaletteIndex(dst.pal, ORD(mode.col[r]), ORD(mode.col[g]), ORD(mode.col[b])));
  2599. mode.transfer := A1OverConst16
  2600. | bgr555, bgr565, bgr466: dst.pack(dst, ADDRESSOF(mode.buf[0]), 0, mode.col); mode.transfer := A1OverConst16
  2601. | bgr888: mode.buf := mode.col; mode.transfer := A1OverConst24
  2602. ELSE mode.transfer := A1OverAny
  2603. END
  2604. | a8: mode.buf := mode.col; mode.transfer := A8OverAny
  2605. | bgra8888:
  2606. CASE dst.code OF
  2607. | bgr555, bgr466, p16: mode.transfer := BGRA8888OverAny16
  2608. | bgr565 : IF MMXenabled THEN
  2609. mode.transfer := MMXBGRA8888Over565;
  2610. IF SSE2enabled THEN mode.transfer := SSE2BGRA8888Over565; END;
  2611. ELSE mode.transfer := BGRA8888Over565
  2612. END
  2613. ELSE mode.transfer := BGRA8888OverAny
  2614. END
  2615. ELSE
  2616. mode.transfer := AnyBlendAny
  2617. END
  2618. END
  2619. ELSE
  2620. mode.transfer := AnyBlendAny
  2621. END;
  2622. ASSERT(mode.transfer # NIL, 120)
  2623. END Bind;
  2624. (**--- Image Operations ---**)
  2625. (** get pixel from image **)
  2626. PROCEDURE Get* (img: Image; x, y: LONGINT; VAR pix: Pixel; VAR mode: Mode);
  2627. VAR bit: LONGINT; adr: ADDRESS;
  2628. BEGIN
  2629. ASSERT((0 <= x) & (x < img.width) & (0 <= y) & (y < img.height), 100);
  2630. bit := x * img.fmt.bpp; adr := img.adr + y * img.bpr + bit DIV 8; bit := bit MOD 8;
  2631. Bind(mode, img.fmt, PixelFormat);
  2632. (*mode.transfer(mode, adr, bit, ADDRESSOF(pix), 0, 1)*)
  2633. mode.transfer(mode, adr, bit, ADDRESSOF(pix[0]), 0, 1) (*PH090122*)
  2634. END Get;
  2635. (** put pixel into image **)
  2636. PROCEDURE Put* (img: Image; x, y: LONGINT; pix: Pixel; VAR mode: Mode);
  2637. VAR bit: LONGINT; adr: ADDRESS;
  2638. BEGIN
  2639. (*ASSERT((0 <= x) & (x < img.width) & (0 <= y) & (y < img.height), 100);*) (*avoid a HALT if roundoff errors from higher levels occur here*)
  2640. IF (0 > x) OR (x >= img.width) OR (0 > y) & (y >= img.height) THEN RETURN END;
  2641. bit := x * img.fmt.bpp; adr := img.adr + y * img.bpr + bit DIV 8; bit := bit MOD 8;
  2642. Bind(mode, PixelFormat, img.fmt);
  2643. mode.transfer(mode, ADDRESSOF(pix[0]), 0, adr, bit, 1)
  2644. END Put;
  2645. (** fill rectangular area **)
  2646. PROCEDURE Fill* (img: Image; llx, lly, urx, ury: LONGINT; pix: Pixel; VAR mode: Mode);
  2647. VAR bit, bb, x, c, t: LONGINT; m: Mode; adr, aa: ADDRESS;
  2648. BEGIN
  2649. ASSERT((0 <= llx) & (llx < urx) & (urx <= img.width) & (0 <= lly) & (lly < ury) & (ury <= img.height), 100);
  2650. bit := llx * img.fmt.bpp; adr := img.adr + lly * img.bpr + bit DIV 8; bit := bit MOD 8;
  2651. IF (mode.op = srcCopy) & (img.fmt.code IN {bgr565}) THEN (* shortcut for speed in important cases *)
  2652. c := ASH(ORD(pix[b]), -3) + ASH(ASH(ORD(pix[g]), -2), 5) + ASH(ASH(ORD(pix[r]), -3), 11);
  2653. t := urx - llx;
  2654. WHILE lly < ury DO
  2655. Fill16(adr, t, c);
  2656. INC(lly); INC(adr, img.bpr)
  2657. END
  2658. ELSIF (mode.op = srcCopy) & (img.fmt.code = bgra8888) THEN (*!to do: move Fill32 with standardized parameters in to Bind() instead*)
  2659. c := SYSTEM.VAL(LONGINT, pix);
  2660. t := urx - llx;
  2661. WHILE lly < ury DO
  2662. Fill32(adr, t, c);
  2663. INC(lly); INC(adr, img.bpr)
  2664. END
  2665. ELSE
  2666. Bind(mode, PixelFormat, img.fmt);
  2667. IF (mode.op IN {clear, srcCopy}) OR (pix[a] = 0FFX) & (mode.op IN {srcOverDst, dstWithoutSrc}) THEN (* dst is replaced *)
  2668. (* copy one pixel to lower left corner of rect *)
  2669. mode.transfer(mode, ADDRESSOF(pix[0]), 0, adr, bit, 1);
  2670. (* copy pixel to rest of bottom row *)
  2671. InitMode(m, srcCopy); Bind(m, img.fmt, img.fmt);
  2672. IF (bit = 0) & (img.fmt.bpp MOD 8 = 0) THEN (* use simple address calculation *)
  2673. bb := img.fmt.bpp DIV 8; aa := adr + bb; x := llx+1;
  2674. WHILE x < urx DO
  2675. m.transfer(m, adr, 0, aa, 0, 1);
  2676. INC(aa, bb); INC(x)
  2677. END
  2678. ELSE
  2679. bb := bit + img.fmt.bpp; aa := adr + bb DIV 8; bb := bb MOD 8; x := llx+1;
  2680. WHILE x < urx DO
  2681. m.transfer(m, adr, bit, aa, bb, 1);
  2682. bb := bb + img.fmt.bpp; aa := aa + bb DIV 8; bb := bb MOD 8; INC(x)
  2683. END
  2684. END;
  2685. (* now copy bottom row to others *)
  2686. INC(lly); aa := adr + img.bpr;
  2687. WHILE lly < ury DO
  2688. m.transfer(m, adr, bit, aa, bit, urx - llx);
  2689. INC(lly); INC(aa, img.bpr)
  2690. END
  2691. ELSE (* fill pixel by pixel *)
  2692. WHILE lly < ury DO
  2693. x := llx; aa := adr; bb := bit;
  2694. WHILE x < urx DO
  2695. mode.transfer(mode, ADDRESSOF(pix[0]), 0, aa, bb, 1);
  2696. bb := bb + img.fmt.bpp; aa := aa + bb DIV 8; bb := bb MOD 8; INC(x)
  2697. END;
  2698. INC(lly); INC(adr, img.bpr)
  2699. END
  2700. END
  2701. END
  2702. END Fill;
  2703. (** clear image **)
  2704. PROCEDURE Clear* (img: Image);
  2705. VAR mode: Mode;
  2706. BEGIN
  2707. InitMode(mode, clear);
  2708. Bind(mode, PixelFormat, img.fmt);
  2709. Fill(img, 0, 0, img.width, img.height, Zero, mode)
  2710. END Clear;
  2711. (** get several pixels and store them in array in requested format **)
  2712. PROCEDURE GetPixels* (img: Image; x, y, w: LONGINT; VAR fmt: Format; VAR buf: ARRAY OF CHAR; ofs : LONGINT; VAR mode: Mode);
  2713. VAR sbit: LONGINT; sadr: ADDRESS;
  2714. BEGIN
  2715. ASSERT((0 <= x) & (x + w <= img.width) & (0 <= y) & (y <= img.height), 100);
  2716. ASSERT(ofs + w * fmt.bpp DIV 8 <= LEN(buf), 101);
  2717. Bind(mode, img.fmt, fmt);
  2718. sbit := x * img.fmt.bpp; sadr := img.adr + y * img.bpr + sbit DIV 8; sbit := sbit MOD 8;
  2719. mode.transfer(mode, sadr, sbit, ADDRESSOF(buf[ofs]), 0, w)
  2720. END GetPixels;
  2721. (** put several pixels from array in given format into image **)
  2722. PROCEDURE PutPixels* (img: Image; x, y, w: LONGINT; VAR fmt: Format; VAR buf: ARRAY OF CHAR; ofs : LONGINT; VAR mode: Mode);
  2723. VAR dbit: LONGINT; dadr: ADDRESS;
  2724. BEGIN
  2725. ASSERT((0 <= x) & (x + w <= img.width) & (0 <= y) & (y <= img.height), 100);
  2726. ASSERT(ofs + w * fmt.bpp DIV 8 <= LEN(buf), 101);
  2727. dbit := x * img.fmt.bpp; dadr := img.adr + y * img.bpr + dbit DIV 8; dbit := dbit MOD 8;
  2728. Bind(mode, fmt, img.fmt);
  2729. mode.transfer(mode, ADDRESSOF(buf[ofs]), 0, dadr, dbit, w)
  2730. END PutPixels;
  2731. (** copy rectangular area to the same or another image in specified mode **)
  2732. PROCEDURE Copy* (src, dst: Image; llx, lly, urx, ury, dx, dy: LONGINT; VAR mode: Mode);
  2733. VAR w, h, sbit, dbit, slen, sb, db, len, l, w1, h1: LONGINT; sadr, dadr, sa, da: ADDRESS;
  2734. BEGIN
  2735. ASSERT((0 <= llx) & (llx <= urx) & (urx <= src.width) & (0 <= lly) & (lly <= ury) & (ury <= src.height), 100);
  2736. ASSERT((0 <= dx) & (dx + urx - llx <= dst.width) & (0 <= dy) & (dy + ury - lly <= dst.height), 101);
  2737. Bind(mode, src.fmt, dst.fmt);
  2738. w := urx - llx; h := ury - lly;
  2739. IF (src # dst) OR (lly > dy) OR (lly = dy) & ((llx > dx) OR (urx <= dx)) THEN (* copy lines bottom-up *)
  2740. sbit := llx * src.fmt.bpp; sadr := src.adr + lly * src.bpr + sbit DIV 8; sbit := sbit MOD 8;
  2741. dbit := dx * dst.fmt.bpp; dadr := dst.adr + dy * dst.bpr + dbit DIV 8; dbit := dbit MOD 8;
  2742. WHILE h > 0 DO
  2743. mode.transfer(mode, sadr, sbit, dadr, dbit, w);
  2744. INC(sadr, src.bpr); INC(dadr, dst.bpr); DEC(h)
  2745. END
  2746. ELSIF lly < dy THEN (* copy lines top-down *)
  2747. sbit := llx * src.fmt.bpp; sadr := src.adr + ury * src.bpr + sbit DIV 8; sbit := sbit MOD 8;
  2748. dbit := dx * dst.fmt.bpp; dadr := dst.adr + (dy + h) * dst.bpr + dbit DIV 8; dbit := dbit MOD 8;
  2749. WHILE h > 0 DO
  2750. DEC(sadr, src.bpr); DEC(dadr, dst.bpr); DEC(h);
  2751. mode.transfer(mode, sadr, sbit, dadr, dbit, w)
  2752. END
  2753. ELSIF llx # dx THEN (* uh oh! overlapping spans *) (* could use CopyN32 with reversed direction flag CLD, see http://en.wikibooks.org/wiki/X86_Assembly/Data_Transfer#Move_String *)
  2754. slen := dx + w - urx; (* maximal span length guaranteeing non-overlapping spans *)
  2755. sbit := urx * src.fmt.bpp; sadr := src.adr + lly * src.bpr + sbit DIV 8; sbit := sbit MOD 8;
  2756. dbit := (dx + w) * dst.fmt.bpp; dadr := dst.adr + dy * dst.bpr + dbit DIV 8; dbit := dbit MOD 8;
  2757. WHILE h > 0 DO
  2758. sa := sadr; sb := sbit; da := dadr; db := dbit; len := w;
  2759. WHILE len > 0 DO
  2760. l := slen;
  2761. IF l > len THEN l := len END;
  2762. DEC(sb, l * src.fmt.bpp); INC(sa, sb DIV 8); sb := sb MOD 8;
  2763. DEC(db, l * dst.fmt.bpp); INC(da, db DIV 8); db := db MOD 8;
  2764. mode.transfer(mode, sa, sb, da, db, l);
  2765. DEC(len, l)
  2766. END;
  2767. INC(sadr, src.bpr); INC(dadr, dst.bpr); DEC(h)
  2768. END
  2769. END
  2770. END Copy;
  2771. (** replicate pattern within rectangular area of image using given mode **)
  2772. PROCEDURE FillPattern* (pat, dst: Image; llx, lly, urx, ury, px, py: LONGINT; VAR mode: Mode);
  2773. VAR pw, ph, olx, oby, ilx, olw, irw, dy, sy, dx, sx, ty: LONGINT;
  2774. BEGIN
  2775. ASSERT((0 <= llx) & (llx <= urx) & (urx <= dst.width) & (0 <= lly) & (lly <= ury) & (ury <= dst.height), 100);
  2776. pw := pat.width; ph := pat.height;
  2777. olx := px + (llx - px) DIV pw * pw;
  2778. oby := py + (lly - py) DIV ph * ph;
  2779. ilx := olx + pw; olw := llx - olx;
  2780. irw := (urx - px) MOD pw;
  2781. IF urx - irw < ilx THEN irw := olw + urx - llx END;
  2782. dy := lly; sy := lly - oby;
  2783. IF (oby < lly) & (oby + ph <= ury) THEN
  2784. dx := llx; sx := olw;
  2785. IF (olx < llx) & (ilx <= urx) THEN
  2786. Copy(pat, dst, sx, sy, pw, ph, llx, lly, mode);
  2787. dx := ilx; sx := 0
  2788. END;
  2789. WHILE dx + pw <= urx DO
  2790. Copy(pat, dst, 0, sy, pw, ph, dx, lly, mode);
  2791. INC(dx, pw)
  2792. END;
  2793. IF dx < urx THEN
  2794. Copy(pat, dst, sx, sy, irw, ph, dx, lly, mode)
  2795. END;
  2796. dy := oby + ph; sy := 0
  2797. END;
  2798. WHILE dy + ph <= ury DO
  2799. dx := llx; sx := olw;
  2800. IF (olx < llx) & (ilx <= urx) THEN
  2801. Copy(pat, dst, sx, 0, pw, ph, llx, dy, mode);
  2802. dx := ilx; sx := 0
  2803. END;
  2804. WHILE dx + pw <= urx DO
  2805. Copy(pat, dst, 0, 0, pw, ph, dx, dy, mode);
  2806. INC(dx, pw)
  2807. END;
  2808. IF dx < urx THEN
  2809. Copy(pat, dst, sx, 0, irw, ph, dx, dy, mode)
  2810. END;
  2811. INC(dy, ph)
  2812. END;
  2813. IF dy < ury THEN
  2814. ty := sy + ury - dy;
  2815. dx := llx; sx := olw;
  2816. IF (olx < llx) & (ilx <= urx) THEN
  2817. Copy(pat, dst, sx, sy, pw, ty, llx, dy, mode);
  2818. dx := ilx; sx := 0
  2819. END;
  2820. WHILE dx + pw <= urx DO
  2821. Copy(pat, dst, 0, sy, pw, ty, dx, dy, mode);
  2822. INC(dx, pw)
  2823. END;
  2824. IF dx < urx THEN
  2825. Copy(pat, dst, sx, sy, irw, ty, dx, dy, mode)
  2826. END
  2827. END
  2828. END FillPattern;
  2829. (** darken image while maintaining coverage **)
  2830. PROCEDURE Darken* (img: Image; factor: REAL);
  2831. VAR s, i, j, k, y, x, bit: LONGINT; adr: ADDRESS; clamp: ARRAY 256 OF CHAR; pix: Pixel;
  2832. BEGIN
  2833. s := ABS(ENTIER(255*factor + 0.5));
  2834. IF (s # 255) & (img.fmt.components # {alpha}) THEN
  2835. i := 256; j := 256*s;
  2836. REPEAT
  2837. DEC(i); DEC(j, s); k := j DIV 255;
  2838. IF k <= 255 THEN clamp[i] := CHR(k) ELSE clamp[i] := 0FFX END
  2839. UNTIL i = 0;
  2840. y := 0;
  2841. WHILE y < img.height DO
  2842. x := 0; adr := img.adr + y * img.bpr; bit := 0;
  2843. WHILE x < img.width DO
  2844. img.fmt.unpack(img.fmt, adr, bit, pix);
  2845. pix[r] := clamp[ORD(pix[r])]; pix[g] := clamp[ORD(pix[g])]; pix[b] := clamp[ORD(pix[b])];
  2846. img.fmt.pack(img.fmt, adr, bit, pix);
  2847. bit := bit + img.fmt.bpp; INC(adr, bit); bit := bit MOD 8;
  2848. INC(x)
  2849. END;
  2850. INC(y)
  2851. END
  2852. END
  2853. END Darken;
  2854. (** fade image **)
  2855. PROCEDURE Fade* (img: Image; factor: REAL);
  2856. VAR s, i, j, k, y, x, bit: LONGINT; adr: ADDRESS; clamp: ARRAY 256 OF CHAR; pix: Pixel;
  2857. BEGIN
  2858. s := ABS(ENTIER(255*factor + 0.5));
  2859. IF s = 0 THEN
  2860. Clear(img)
  2861. ELSIF s # 255 THEN
  2862. i := 256; j := 256*s;
  2863. REPEAT
  2864. DEC(i); DEC(j, s); k := j DIV 255;
  2865. IF k <= 255 THEN clamp[i] := CHR(k) ELSE clamp[i] := 0FFX END
  2866. UNTIL i = 0;
  2867. y := 0;
  2868. WHILE y < img.height DO
  2869. x := 0; adr := img.adr + y * img.bpr; bit := 0;
  2870. WHILE x < img.width DO
  2871. img.fmt.unpack(img.fmt, adr, bit, pix);
  2872. pix[r] := clamp[ORD(pix[r])]; pix[g] := clamp[ORD(pix[g])];
  2873. pix[b] := clamp[ORD(pix[b])]; pix[a] := clamp[ORD(pix[a])];
  2874. img.fmt.pack(img.fmt, adr, bit, pix);
  2875. bit := bit + img.fmt.bpp; INC(adr, bit); bit := bit MOD 8;
  2876. INC(x)
  2877. END;
  2878. INC(y)
  2879. END
  2880. END
  2881. END Fade;
  2882. (** make image brighter and more transparent; Opaque(I, f) = Darken(Fade(I, f), 1/f) **)
  2883. PROCEDURE Opaque* (img: Image; factor: REAL);
  2884. VAR s, i, j, k, y, x, bit: LONGINT; adr: ADDRESS; clamp: ARRAY 256 OF CHAR; pix: Pixel;
  2885. BEGIN
  2886. s := ABS(ENTIER(255*factor + 0.5));
  2887. IF s = 0 THEN
  2888. Clear(img)
  2889. ELSIF s # 255 THEN
  2890. i := 256; j := 256*s;
  2891. REPEAT
  2892. DEC(i); DEC(j, s); k := j DIV 255;
  2893. IF k <= 255 THEN clamp[i] := CHR(k) ELSE clamp[i] := 0FFX END
  2894. UNTIL i = 0;
  2895. y := 0;
  2896. WHILE y < img.height DO
  2897. x := 0; adr := img.adr + y * img.bpr; bit := 0;
  2898. WHILE x < img.width DO
  2899. img.fmt.unpack(img.fmt, adr, bit, pix);
  2900. pix[a] := clamp[ORD(pix[a])];
  2901. img.fmt.pack(img.fmt, adr, bit, pix);
  2902. bit := bit + img.fmt.bpp; INC(adr, bit); bit := bit MOD 8;
  2903. INC(x)
  2904. END;
  2905. INC(y)
  2906. END
  2907. END
  2908. END Opaque;
  2909. (** add components of two (faded) images **)
  2910. PROCEDURE Add* (i, j, res: Image);
  2911. VAR y, x, ibit, jbit, rbit, k: LONGINT; iadr, jadr, radr: ADDRESS; ipix, jpix, rpix: Pixel;
  2912. BEGIN
  2913. ASSERT((i.width = j.width) & (i.height = j.height) & (i.width <= res.width) & (i.height <= res.height), 100);
  2914. y := 0;
  2915. WHILE y < i.height DO
  2916. x := 0; iadr := i.adr + y * i.bpr; ibit := 0; jadr := j.adr + y * j.bpr; jbit := 0; radr := res.adr + y * res.bpr; rbit := 0;
  2917. WHILE x < i.width DO
  2918. i.fmt.unpack(i.fmt, iadr, ibit, ipix); j.fmt.unpack(j.fmt, jadr, jbit, jpix);
  2919. FOR k := 0 TO 3 DO
  2920. rpix[k] := Clamp[ORD(ipix[k]) + ORD(jpix[k])]
  2921. END;
  2922. res.fmt.pack(res.fmt, radr, rbit, rpix);
  2923. ibit := ibit + i.fmt.bpp; INC(iadr, ibit); ibit := ibit MOD 8;
  2924. jbit := jbit + j.fmt.bpp; INC(jadr, jbit); jbit := jbit MOD 8;
  2925. rbit := rbit + res.fmt.bpp; INC(radr, rbit); rbit := rbit MOD 8;
  2926. INC(x)
  2927. END;
  2928. INC(y)
  2929. END
  2930. END Add;
  2931. (** copy image to another using error diffusion dithering (Floyd-Steinberg) **)
  2932. PROCEDURE Dither* (src, dst: Image);
  2933. TYPE
  2934. error = RECORD r, g, b: LONGINT END;
  2935. VAR
  2936. e351: POINTER TO ARRAY OF error;
  2937. y, x, sb, db, ex, e, e3, e5: LONGINT;
  2938. sadr, dadr, sa, da: ADDRESS;
  2939. e7, e51, e1: error;
  2940. spix, dpix: Pixel;
  2941. BEGIN
  2942. ASSERT((src.width <= dst.width) & (src.height <= dst.height), 100);
  2943. NEW(e351, src.width+2); (* accumulated error for next row *)
  2944. y := 0; sadr := src.adr; dadr := dst.adr;
  2945. WHILE y < src.height DO (* scan from left to right *)
  2946. e7.r := 0; e7.g := 0; e7.b := 0;
  2947. e51.r := 0; e51.g := 0; e51.b := 0;
  2948. e1.r := 0; e1.g := 0; e1.b := 0;
  2949. x := 0; sa := sadr; sb := 0; da := dadr; db := 0;
  2950. WHILE x < src.width DO
  2951. ex := x+1;
  2952. src.fmt.unpack(src.fmt, sa, sb, spix);
  2953. spix[r] := Clamp[200H + ORD(spix[r]) + e351[ex].r + e7.r];
  2954. spix[g] := Clamp[200H + ORD(spix[g]) + e351[ex].g + e7.g];
  2955. spix[b] := Clamp[200H + ORD(spix[b]) + e351[ex].b + e7.b];
  2956. dst.fmt.pack(dst.fmt, da, db, spix);
  2957. dst.fmt.unpack(dst.fmt, da, db, dpix);
  2958. e := ORD(spix[r]) - ORD(dpix[r]); e3 := 3*e DIV 16; e5 := 5*e DIV 16;
  2959. e7.r := 7*e DIV 16; e351[x].r := e3 + e51.r; e51.r := e5 + e1.r; e1.r := e - e3 - e5 - e7.r;
  2960. e := ORD(spix[g]) - ORD(dpix[g]); e3 := 3*e DIV 16; e5 := 5*e DIV 16;
  2961. e7.g := 7*e DIV 16; e351[x].g := e3 + e51.g; e51.g := e5 + e1.g; e1.g := e - e3 - e5 - e7.g;
  2962. e := ORD(spix[b]) - ORD(dpix[b]); e3 := 3*e DIV 16; e5 := 5*e DIV 16;
  2963. e7.b := 7*e DIV 16; e351[x].b := e3 + e51.b; e51.b := e5 + e1.b; e1.b := e - e3 - e5 - e7.b;
  2964. sb := sb + src.fmt.bpp; INC(sa, sb DIV 8); sb := sb MOD 8;
  2965. db := db + dst.fmt.bpp; INC(da, db DIV 8); db := db MOD 8;
  2966. x := ex
  2967. END;
  2968. INC(y); INC(sadr, src.bpr); INC(dadr, dst.bpr);
  2969. IF y < src.height THEN (* scan from right to left *)
  2970. e351[x] := e51;
  2971. e7.r := 0; e7.g := 0; e7.b := 0;
  2972. e51.r := 0; e51.g := 0; e51.b := 0;
  2973. e1.r := 0; e1.g := 0; e1.b := 0;
  2974. INC(sa, src.bpr); INC(da, dst.bpr);
  2975. WHILE x > 0 DO
  2976. ex := x; DEC(x);
  2977. sb := sb - src.fmt.bpp; INC(sa, sb DIV 8); sb := sb MOD 8;
  2978. db := db - dst.fmt.bpp; INC(da, db DIV 8); db := db MOD 8;
  2979. src.fmt.unpack(src.fmt, sa, sb, spix);
  2980. spix[r] := Clamp[200H + ORD(spix[r]) + e351[ex].r + e7.r];
  2981. spix[g] := Clamp[200H + ORD(spix[g]) + e351[ex].g + e7.g];
  2982. spix[b] := Clamp[200H + ORD(spix[b]) + e351[ex].b + e7.b];
  2983. dst.fmt.pack(dst.fmt, da, db, spix);
  2984. dst.fmt.unpack(dst.fmt, da, db, dpix);
  2985. INC(ex);
  2986. e := ORD(spix[r]) - ORD(dpix[r]); e3 := 3*e DIV 16; e5 := 5*e DIV 16;
  2987. e7.r := 7*e DIV 16; e351[x].r := e3 + e51.r; e51.r := e5 + e1.r; e1.r := e - e3 - e5 - e7.r;
  2988. e := ORD(spix[g]) - ORD(dpix[g]); e3 := 3*e DIV 16; e5 := 5*e DIV 16;
  2989. e7.g := 7*e DIV 16; e351[x].g := e3 + e51.g; e51.g := e5 + e1.g; e1.g := e - e3 - e5 - e7.g;
  2990. e := ORD(spix[b]) - ORD(dpix[b]); e3 := 3*e DIV 16; e5 := 5*e DIV 16;
  2991. e7.b := 7*e DIV 16; e351[x].b := e3 + e51.b; e51.b := e5 + e1.b; e1.b := e - e3 - e5 - e7.b
  2992. END;
  2993. e351[1] := e51;
  2994. INC(y); INC(sadr, src.bpr); INC(dadr, dst.bpr)
  2995. END
  2996. END
  2997. END Dither;
  2998. (**--- File I/O ---**)
  2999. (** write image to file rider **)
  3000. PROCEDURE Write* (VAR fr: Streams.Writer; img: Image);
  3001. VAR m: Image; h, w, len: LONGINT; adr, aa: ADDRESS; buf: ARRAY 256 OF CHAR;
  3002. SrcCopy:Mode;
  3003. BEGIN
  3004. InitMode(SrcCopy, srcCopy);
  3005. IF ~(img.fmt.code IN {a1..bgra8888,p16}) THEN
  3006. NEW(m);
  3007. IF img.fmt.components = {color} THEN Create(m, img.width, img.height, BGR888)
  3008. ELSIF img.fmt.components = {alpha} THEN Create(m, img.width, img.height, A8)
  3009. ELSIF img.fmt.components = {index} THEN Create(m, img.width, img.height, D8)
  3010. ELSE Create(m, img.width, img.height, BGRA8888)
  3011. END;
  3012. Copy(img, m, 0, 0, img.width, img.height, 0, 0, SrcCopy);
  3013. img := m
  3014. END;
  3015. fr.RawNum(2); (* version *)
  3016. fr.RawNum(img.fmt.code);
  3017. fr.RawNum(img.width); fr.RawNum(img.height);
  3018. fr.RawNum(ABS(img.bpr));
  3019. h := img.height; adr := img.adr;
  3020. WHILE h > 0 DO
  3021. w := ABS(img.bpr); aa := adr;
  3022. WHILE w > 0 DO
  3023. len := 256;
  3024. IF len > w THEN len := w END;
  3025. SYSTEM.MOVE(aa, ADDRESSOF(buf[0]), len);
  3026. fr.Bytes(buf, 0, len);
  3027. DEC(w, len); INC(aa, len)
  3028. END;
  3029. DEC(h); INC(adr, img.bpr)
  3030. END;
  3031. IF img.fmt.code IN {p8,p16} THEN
  3032. fr.RawNum(img.fmt.pal.used);
  3033. len := 0;
  3034. WHILE len < img.fmt.pal.used DO
  3035. fr.Char(img.fmt.pal.col[len, r]);
  3036. fr.Char(img.fmt.pal.col[len, g]);
  3037. fr.Char(img.fmt.pal.col[len, b]);
  3038. INC(len)
  3039. END
  3040. END;
  3041. fr.Update (* optional *)
  3042. END Write;
  3043. (** read image from file rider **)
  3044. PROCEDURE Read* (VAR fr: Streams.Reader; img: Image);
  3045. VAR ver, code, w, h, bpr, len, bytesRead: LONGINT; adr, aa: ADDRESS; fmt: Format; buf: ARRAY 256 OF CHAR; used: LONGINT;
  3046. BEGIN
  3047. ASSERT(img#NIL,100);
  3048. fr.RawNum(ver); (* know version 1&2*)
  3049. ASSERT(ver IN {1,2},101);
  3050. fr.RawNum(code);
  3051. CASE code OF
  3052. | a1: fmt := A1
  3053. | a8: fmt := A8
  3054. | p8: InitFormat(fmt, p8, 8, 1, {index}, NIL, PackP8, UnpackP8)
  3055. | d8: fmt := D8
  3056. | p16: InitFormat(fmt, p16, 16, 2, {index}, NIL, PackP16, UnpackP16);
  3057. | bgr555: fmt := BGR555
  3058. | bgr565: fmt := BGR565
  3059. | bgr466: fmt := BGR466
  3060. | bgr888: fmt := BGR888
  3061. | bgra8888: fmt := BGRA8888
  3062. END;
  3063. fr.RawNum(w); fr.RawNum(h);
  3064. Create(img, SHORT(w), SHORT(h), fmt);
  3065. fr.RawNum(bpr);
  3066. ASSERT(bpr <= img.bpr);
  3067. adr := img.adr;
  3068. WHILE h > 0 DO
  3069. w := bpr; aa := adr;
  3070. WHILE w > 0 DO
  3071. len := 256;
  3072. IF len > w THEN len := w END;
  3073. fr.Bytes(buf, 0, len, bytesRead); (* ignore bytesRead *)
  3074. SYSTEM.MOVE(ADDRESSOF(buf[0]), aa, len);
  3075. DEC(w, len); INC(aa, len)
  3076. END;
  3077. DEC(h); INC(adr, img.bpr)
  3078. END;
  3079. IF code IN {p8,p16} THEN
  3080. fr.RawNum(used);
  3081. len := 0;
  3082. NEW(img.fmt.pal); img.fmt.pal.Init(used);
  3083. WHILE len < used DO
  3084. fr.Char(img.fmt.pal.col[len, r]);
  3085. fr.Char(img.fmt.pal.col[len, g]);
  3086. fr.Char(img.fmt.pal.col[len, b]);
  3087. img.fmt.pal.col[len, a] := 0FFX;
  3088. INC(len)
  3089. END;
  3090. InitPalette(img.fmt.pal, used, 4)
  3091. END
  3092. END Read;
  3093. (*--- Initialization ---*)
  3094. PROCEDURE InitBitTables;
  3095. VAR b, i: LONGINT;
  3096. BEGIN
  3097. FOR b := 0 TO 0FFH DO
  3098. FOR i := 0 TO 7 DO
  3099. IF ODD(ASH(b, -i)) THEN
  3100. Bit[b, i] := TRUE; Set[b, i] := CHR(b); Clr[b, i] := CHR(b - ASH(1, i))
  3101. ELSE
  3102. Bit[b, i] := FALSE; Set[b, i] := CHR(b + ASH(1, i)); Clr[b, i] := CHR(b)
  3103. END
  3104. END
  3105. END
  3106. END InitBitTables;
  3107. PROCEDURE InitClamp;
  3108. VAR i: LONGINT;
  3109. BEGIN
  3110. FOR i := 0 TO 1FFH DO Clamp[i] := 0X END;
  3111. FOR i := 0 TO 0FFH DO Clamp[200H+i] := CHR(i) END;
  3112. FOR i := 300H TO 4FFH DO Clamp[i] := 0FFX END
  3113. END InitClamp;
  3114. PROCEDURE ToggleMMX*;
  3115. BEGIN
  3116. MMXenabled := ~MMXenabled
  3117. END ToggleMMX;
  3118. PROCEDURE ToggleSSE2*;
  3119. BEGIN
  3120. SSE2enabled := ~SSE2enabled;
  3121. KernelLog.String("SSE2 toggled! Is now: "); KernelLog.Boolean(SSE2enabled);KernelLog.Ln;
  3122. END ToggleSSE2;
  3123. (** Map a color value to an 8-bit CLUT index. Only used if format = index8. *)
  3124. PROCEDURE ColorToIndex*(col: LONGINT): LONGINT;
  3125. BEGIN
  3126. (* default implementation is not very useful and should be overridden. *)
  3127. RETURN SYSTEM.VAL(LONGINT,
  3128. SYSTEM.VAL(SET, ASH(col, 7-23)) * {5..7} +
  3129. SYSTEM.VAL(SET, ASH(col, 4-15)) * {2..4} +
  3130. SYSTEM.VAL(SET, ASH(col, 1-7)) * {0..1})
  3131. END ColorToIndex;
  3132. (** Map an 8-bit CLUT index to a color value. Only used if format = index8. *)
  3133. PROCEDURE IndexToColor*(index: LONGINT): LONGINT;
  3134. BEGIN
  3135. (* default implementation is not very useful and should be overridden. *)
  3136. RETURN
  3137. ASH(SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, index) * {5..7}), 23-7) +
  3138. ASH(SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, index) * {2..4}), 15-4) +
  3139. ASH(SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, index) * {0..1}), 7-1)
  3140. END IndexToColor;
  3141. BEGIN
  3142. MMXenabled := 23 IN Machine.features;
  3143. SSE2enabled := Machine.SSE2Support;
  3144. (* plugin := Displays.registry.Await(""); (* assume only one system-wide D8 display driver *)
  3145. d8display := plugin(Displays.Display); *)
  3146. InitFormat(A1, a1, 1, 1, {alpha}, NIL, PackA1, UnpackA1);
  3147. InitFormat(A8, a8, 8, 1, {alpha}, NIL, PackA8, UnpackA8);
  3148. InitFormat(D8, d8, 8, 1, {index}, NIL, PackD8, UnpackD8);
  3149. InitFormat(BGR555, bgr555, 16, 2, {color}, NIL, PackBGR555, UnpackBGR555);
  3150. InitFormat(BGR565, bgr565, 16, 2, {color}, NIL, PackBGR565, UnpackBGR565);
  3151. InitFormat(BGR466, bgr466, 16, 2, {color}, NIL, PackBGR466, UnpackBGR466);
  3152. InitFormat(BGR888, bgr888, 24, 4, {color}, NIL, PackBGR888, UnpackBGR888);
  3153. InitFormat(BGRA8888, bgra8888, 32, 4, {color, alpha}, NIL, PackBGRA8888, UnpackBGRA8888);
  3154. PixelFormat := BGRA8888;
  3155. Zero[0] := 0X; Zero[1] := 0X; Zero[2] := 0X; Zero[3] := 0X;
  3156. InitBitTables; InitClamp
  3157. END Raster.
  3158. (**
  3159. Remarks
  3160. 1. Images
  3161. While many applications wish to handle images of any kind without having to care about details, other applications need low-level access to image interna for maximum effiency. With this in mind, the Images module provides an abstract procedural interface but also discloses low-level information to those clients needing it:
  3162. * an image references a contiguous block of memory holding pixel data
  3163. * the point of reference is the address of the pixel in the lower-left corner
  3164. * pixels are organized in rows (either bottom-up or top-down)
  3165. * rows can be aligned to an arbitrary number of bytes
  3166. * the leftmost pixel in a row has the lowest address of all pixels in that row
  3167. * every pixel uses the same number of bits
  3168. Memory for images can be automatically allocated by using Create(). Alternatively, an image can be initialized on an existing memory block (Init(), InitBuf()) or even on part of an other image (InitRect()).
  3169. 2. Pixels
  3170. A general pixel pix[] contains four components (in range 0X..255X), specifying red, green, blue, and alpha value of the pixel and accessable as pix[r], pix[g], pix[b] and pix[a]. Note that in order to speed up compositing operations, the alpha value is premultiplied into the color components. Example: a red pixel with 50% coverage can be initialized with SetRGBA(pix, 255, 0, 0, 127), after which pix[r]=pix[a]=7FX and pix[g]=pix[b]=0X. Use GetRGBA() to recover the original color and alpha values.
  3171. 3. Palettes
  3172. Many bitmap images and Oberon display drivers use some kind of indexed format to store colors, i.e. the value stored in the bitmap serves as an index into an array of colors. A Palette stores up to 256 colors as an array of pixels, making the mapping of an index to the corresponding color straightforward. To speed up the inverse mapping from an RGB triple to an index with PaletteIndex(), additional data is initialized when InitPalette() is called. Use ComputePalette() to compute a palette that best approximates the colors in a given image (e.g. before quantizing it to indexed format).
  3173. 4. Formats
  3174. While general pixels accurately describe color and alpha information, they use a lot of memory (32 bits). Most images therefore only store part of that information. A Format record describes how pixels are represented within an image. It contains
  3175. * the number of bits used per pixel (must be 1, 2, 4, 8, 16, 24 or 32)
  3176. * the set of components stored in a pixel (color, index and/or alpha)
  3177. * a palette if the format uses one
  3178. * procedures for storing (packing) and loading (unpacking) a general pixel
  3179. The pack and unpack procedures are given an address and a bit number specifying where the pixel is located in memory, with bit numbers ascending from left to right (although a format is free to choose any bit ordering within a pixel).
  3180. 5. Predefined Formats
  3181. The following global variables contain formats which are special and have a unique code number identifying them. Besides, most operations have better performance if acting on images using them.
  3182. * A1 (code a1): one bit alpha, MSB leftmost (corresponds to Oberon display patterns)
  3183. * A8 (code a8): 8 bit alpha (mainly for anti-aliased character patterns)
  3184. * - (code p8): 8 bit indexed with custom palette (Oberon pictures, use InitPaletteFormat to initialize)
  3185. * D8 (code d8): 8 bit indexed with display palette (no palette structure attached)
  3186. * - (code p16): 16 bit indexed with 16bit Palette. This type is, e.g., often used in medical imaging (DICOM-3 standard) (* PH 2004 *)
  3187. * BGR555 (code bgr555), BGR565 (code bgr565), BGR466 (code bgr466): 16 bit hi-color
  3188. * BGR888 (code bgr888): 24 bit true-color
  3189. * BGRA8888 (code bgra8888), PixelFormat: 32 bit true-color with alpha channel (general pixel format)
  3190. Procedure DisplayFormat() returns the format that best matches the supplied kind of display transfer format. The returned image format is preferably used for allocating shadow bitmaps.
  3191. 6. Compositing
  3192. Most operations require a transfer mode for specifying how source and destination pixels should be combined when alpha information is present. The following compositing operations are supported:
  3193. * clear: destination becomes black and completely transparent
  3194. * srcCopy: source completely replaces destination (cf. Display.replace)
  3195. * dstCopy: no effect
  3196. * srcOverDst: source replaces destination where source is opaque (cf. Display.paint)
  3197. * dstOverSrc: destination replaces source where destination is opaque
  3198. * srcInDst: source where destination is opaque
  3199. * dstInSrc: destination where source is opaque
  3200. * srcWithoutDest*: source is cleared where destination is opaque
  3201. * dstWithoutSrc*: destination is cleared where source is opaque
  3202. * srcAtopDst*: source replaces destination where destination is opaque
  3203. * dstAtopSrc*: destination replaces source where source is opaque
  3204. * srcXorDst*: destination is cleared where both source and destination are opaque (cf. Display.invert)
  3205. A transfer mode is initialized with a compositing operation and optionally with a color. (The color is used when the source is a pure alpha format which doesn't contain any color of its own.) An initialized mode can be bound to a source and destination format by calling Bind(), by which the mode's transfer procedure is set appropriately. A transfer procedure unpacks pixels from source and destination, blends them according to the compositing operation, and packs the resulting pixel in the destination. Bind() chooses an optimal transfer procedure for the given combination of compositing operation, source format, and destination format.
  3206. 7. Internalization and Externalization
  3207. Images can be loaded from file and stored to file using one of many popular image file formats. The Load() and Store() procedures rely on a section 'ImageFormats' in the Aos registry. This section contains a list of file types that are associated with command procedures. When one of these commands is called, it should initialize the global 'LoadProc' and 'StoreProc' variables. These, when called, should read an image from or write an image to the file and set 'done' to TRUE if successful.
  3208. **)