VM Maker: VMMaker.oscog-eem.2457.mcz

Previous Topic Next Topic
 
classic Classic list List threaded Threaded
1 message Options
Reply | Threaded
Open this post in threaded view
|

VM Maker: VMMaker.oscog-eem.2457.mcz

commits-2
 
Eliot Miranda uploaded a new version of VMMaker to project VM Maker:
http://source.squeak.org/VMMaker/VMMaker.oscog-eem.2457.mcz

==================== Summary ====================

Name: VMMaker.oscog-eem.2457
Author: eem
Time: 15 October 2018, 5:07:51.317632 pm
UUID: 18deefc8-1a84-4c5f-ab59-02af686dfc0b
Ancestors: VMMaker.oscog-AlistairGrant.2456

BitBltPlugin/BitBltSimulation

Add asserts and variables for bounds checking destination and source access.

Use the _: style for calling the surface functions.

Mark all memory access methods as <inline: #always> so as not to bother generating rthe uninlined versions.

=============== Diff against VMMaker.oscog-AlistairGrant.2456 ===============

Item was changed:
  SmartSyntaxInterpreterPlugin subclass: #BitBltSimulation
+ instanceVariableNames: 'destForm sourceForm halftoneForm combinationRule destX destY width height sourceX sourceY clipX clipY clipWidth clipHeight sourceWidth sourceHeight sourceDepth sourcePitch sourceBits sourcePPW sourceMSB destWidth destHeight destDepth destPitch destBits destPPW destMSB bitCount skew mask1 mask2 preload nWords destMask hDir vDir sourceIndex sourceDelta destIndex destDelta sx sy dx dy bbW bbH halftoneHeight noSource noHalftone halftoneBase sourceAlpha srcBitShift dstBitShift bitBltOop affectedL affectedR affectedT affectedB opTable maskTable ditherMatrix4x4 ditherThresholds16 ditherValues16 hasSurfaceLock warpSrcShift warpSrcMask warpAlignShift warpAlignMask warpBitShiftTable querySurfaceFn lockSurfaceFn unlockSurfaceFn isWarping cmFlags cmMask cmShiftTable cmMaskTable cmLookupTable cmBitsPerColor dither8Lookup componentAlphaModeColor componentAlphaModeAlpha ungammaLookupTable gammaLookupTable numGCsOnInvocation bitBltIsReceiver endOfDestination endOfSource'
- instanceVariableNames: 'destForm sourceForm halftoneForm combinationRule destX destY width height sourceX sourceY clipX clipY clipWidth clipHeight sourceWidth sourceHeight sourceDepth sourcePitch sourceBits sourcePPW sourceMSB destWidth destHeight destDepth destPitch destBits destPPW destMSB bitCount skew mask1 mask2 preload nWords destMask hDir vDir sourceIndex sourceDelta destIndex destDelta sx sy dx dy bbW bbH halftoneHeight noSource noHalftone halftoneBase sourceAlpha srcBitShift dstBitShift bitBltOop affectedL affectedR affectedT affectedB opTable maskTable ditherMatrix4x4 ditherThresholds16 ditherValues16 hasSurfaceLock warpSrcShift warpSrcMask warpAlignShift warpAlignMask warpBitShiftTable querySurfaceFn lockSurfaceFn unlockSurfaceFn isWarping cmFlags cmMask cmShiftTable cmMaskTable cmLookupTable cmBitsPerColor dither8Lookup componentAlphaModeColor componentAlphaModeAlpha ungammaLookupTable gammaLookupTable numGCsOnInvocation bitBltIsReceiver'
  classVariableNames: 'AllOnes AlphaIndex BBClipHeightIndex BBClipWidthIndex BBClipXIndex BBClipYIndex BBColorMapIndex BBDestFormIndex BBDestXIndex BBDestYIndex BBHalftoneFormIndex BBHeightIndex BBLastIndex BBRuleIndex BBSourceFormIndex BBSourceXIndex BBSourceYIndex BBWarpBase BBWidthIndex BBXTableIndex BEBitBltIndex BinaryPoint BlueIndex ColorMapFixedPart ColorMapIndexedPart ColorMapNewStyle ColorMapPresent CrossedX EndOfRun FixedPt1 FormBitsIndex FormDepthIndex FormHeightIndex FormWidthIndex GreenIndex OpTable OpTableSize RedIndex'
  poolDictionaries: ''
  category: 'VMMaker-Interpreter'!
 
  !BitBltSimulation commentStamp: 'tpr 3/25/2013 16:50' prior: 0!
  This class implements BitBlt, much as specified in the Blue Book spec.
 
  Performance has been enhanced through the use of pointer variables such as sourceIndex and destIndex, and by separating several special cases of the inner loop.
 
  Operation has been extended to color, with support for 1, 2, 4, 8, 16, and 32-bit pixel sizes.  Conversion between different pixel sizes is facilitated by accepting an optional color map.
 
  In addition to the original 16 combination rules, this BitBlt supports
  16 fail (for old paint mode)
  17 fail (for old mask mode)
  18 sourceWord + destinationWord
  19 sourceWord - destinationWord
  20 rgbAdd: sourceWord with: destinationWord
  21 rgbSub: sourceWord with: destinationWord
  22 OLDrgbDiff: sourceWord with: destinationWord
  23 OLDtallyIntoMap: destinationWord -- old vers doesn't clip to bit boundary
  24 alphaBlend: sourceWord with: destinationWord
  25 pixPaint: sourceWord with: destinationWord
  26 pixMask: sourceWord with: destinationWord
  27 rgbMax: sourceWord with: destinationWord
  28 rgbMin: sourceWord with: destinationWord
  29 rgbMin: sourceWord bitInvert32 with: destinationWord
  30 alphaBlendConst: sourceWord with: destinationWord -- alpha passed as an arg
  31 alphaPaintConst: sourceWord with: destinationWord -- alpha passed as an arg
  32 rgbDiff: sourceWord with: destinationWord
  33 tallyIntoMap: destinationWord
  34 alphaBlendScaled: sourceWord with: destinationWord
  35 alphaBlendScaled: sourceWord with: "unused here - only used by FXBlt"
  36 alphaBlendScaled: sourceWord with: "unused here - only used by FXBlt"
  37 rgbMul: sourceWord with: destinationWord
  38 pixSwap: sourceWord with: destinationWord
  39 pixClear: sourceWord with: destinationWord
  40 fixAlpha: sourceWord with: destinationWord
  41 rgbComponentAlpha: sourceWord with: destinationWord
 
  This implementation has also been fitted with an experimental "warp drive" that allows abritrary scaling and rotation (and even limited affine deformations) with all BitBlt storage modes supported.
 
  To add a new rule to BitBlt...
  1.  add the new rule method or methods in the category 'combination rules' of BBSim
  2.  describe it in the class comment  of BBSim and in the class comment for BitBlt
  3.  add refs to initializeRuleTable in proper positions
  4.  add refs to initBBOpTable, following the pattern
  !

Item was changed:
  ----- Method: BitBltSimulation class>>declareCVarsIn: (in category 'translation') -----
  declareCVarsIn: aCCodeGenerator
 
  "add option of  fast path BitBLT code header"
  aCCodeGenerator
  addHeaderFile:'"sqAssert.h"';
  addHeaderFile:'#ifdef ENABLE_FAST_BLT
  #include "BitBltDispatch.h"
  #else
  // to handle the unavoidable decl in the spec of copyBitsFallback();
  #define operation_t void
  #endif'.
 
  aCCodeGenerator var: 'opTable'
  declareC: 'void *opTable[' , OpTableSize printString , ']'.
  aCCodeGenerator var: 'maskTable'
  declareC:'int maskTable[33] = {
  0, 1, 3, 0, 15, 31, 0, 0, 255, 0, 0, 0, 0, 0, 0, 0, 65535,
  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -1
  }'.
  aCCodeGenerator var: 'ditherMatrix4x4'
  declareC:'const int ditherMatrix4x4[16] = {
  0, 8, 2, 10,
  12, 4, 14, 6,
  3, 11, 1, 9,
  15, 7, 13, 5
  }'.
  aCCodeGenerator var: 'ditherThresholds16'
  declareC:'const int ditherThresholds16[8] = { 0, 2, 4, 6, 8, 12, 14, 16 }'.
  aCCodeGenerator var: 'ditherValues16'
  declareC:'const int ditherValues16[32] = {
  0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
  15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30
  }'.
 
  aCCodeGenerator var: 'warpBitShiftTable'
  declareC:'int warpBitShiftTable[32]'.
 
  aCCodeGenerator var:'cmShiftTable'
  type:'int *'.
  aCCodeGenerator var:'cmMaskTable'
  type:'unsigned int *'.
  aCCodeGenerator var:'cmLookupTable'
  type:'unsigned int *'.
 
  aCCodeGenerator var: 'dither8Lookup'
  declareC:' unsigned char dither8Lookup[4096]'.
 
  aCCodeGenerator var:'ungammaLookupTable'
  type: 'unsigned char *'.
  aCCodeGenerator var:'gammaLookupTable'
  type: 'unsigned char *'.
 
  aCCodeGenerator
  var: 'querySurfaceFn' declareC: 'int (*querySurfaceFn)(sqIntptr_t, int*, int*, int*, int*)';
  var: 'lockSurfaceFn' declareC: 'sqIntptr_t (*lockSurfaceFn)(sqIntptr_t, int*, int, int, int, int)';
  var: 'unlockSurfaceFn' declareC: 'int (*unlockSurfaceFn)(sqIntptr_t, int, int, int, int)'.
 
  #(sourcePitch sourceWidth sourceHeight sourceDepth sourceMSB sx sy
  destPitch destWidth destHeight destDepth destMSB dx dy bbW bbH)
+ do: [:ivar | aCCodeGenerator var: ivar type: #int].
+
+ #('sourceIndex' 'destIndex' 'endOfDestination' 'endOfSource') do:
+ [:ivar| aCCodeGenerator var: ivar type: #usqInt]!
- do: [:ivar | aCCodeGenerator var: ivar type: #int]!

Item was changed:
  ----- Method: BitBltSimulation>>dstLongAt: (in category 'memory access') -----
  dstLongAt: idx
+ <inline: #always>
+ self assert: idx asUnsignedInteger < endOfDestination.
-
  ^self long32At: idx!

Item was changed:
  ----- Method: BitBltSimulation>>dstLongAt:put: (in category 'memory access') -----
  dstLongAt: idx put: value
+ <inline: #always>
+ "We omit the assert here since dstLongAt:put: is always preceded by a dstLongAt: which does have an assert.
+ self assert: idx asUnsignedInteger < endOfDestination."
-
  ^self long32At: idx put: value!

Item was changed:
  ----- Method: BitBltSimulation>>dstLongAt:put:mask: (in category 'memory access') -----
  dstLongAt: idx put: srcValue mask: dstMask
  "Store the given value back into destination form, using dstMask
+ to mask out the bits to be modified. This is an essential
+ read-modify-write operation on the destination form."
+ <inline: #always>
- to mask out the bits to be modified. This is an essiantial
- read-modify-write operation on the destination form."
  | dstValue |
- <inline: true>
  dstValue := self dstLongAt: idx.
  dstValue := dstValue bitAnd: dstMask.
  dstValue := dstValue bitOr: srcValue.
  self dstLongAt: idx put: dstValue.!

Item was changed:
  ----- Method: BitBltSimulation>>halftoneAt: (in category 'memory access') -----
  halftoneAt: idx
  "Return a value from the halftone pattern."
+ <inline: #always>
-
  ^self long32At: halftoneBase + (idx \\ halftoneHeight * 4)!

Item was changed:
  ----- Method: BitBltSimulation>>incDestIndex: (in category 'memory access') -----
  incDestIndex: offset
+ <inline: #always>
+ ^destIndex := destIndex + offset
- <inline: true>
- ^ destIndex := destIndex + offset
  !

Item was changed:
  ----- Method: BitBltSimulation>>incSrcIndex: (in category 'memory access') -----
  incSrcIndex: offset
+ <inline: #always>
+ ^sourceIndex := sourceIndex + offset
- <inline: true>
- ^ sourceIndex := sourceIndex + offset
  !

Item was changed:
  ----- Method: BitBltSimulation>>loadBitBltDestForm (in category 'interpreter interface') -----
  loadBitBltDestForm
  "Load the dest form for BitBlt. Answer false if anything is wrong, true otherwise."
 
  | destBitsSize |
  <inline: true>
  ((interpreterProxy isPointers: destForm)
  and: [(interpreterProxy slotSizeOf: destForm) >= 4]) ifFalse:
  [^false].
  destBits := interpreterProxy fetchPointer: FormBitsIndex ofObject: destForm.
  destWidth := interpreterProxy fetchInteger: FormWidthIndex ofObject: destForm.
  destHeight := interpreterProxy fetchInteger: FormHeightIndex ofObject: destForm.
  (destWidth >= 0 and: [destHeight >= 0]) ifFalse:
  [^false].
  destDepth := interpreterProxy fetchInteger: FormDepthIndex ofObject: destForm.
+ (destMSB := destDepth > 0) ifFalse:
- destMSB := destDepth > 0.
- destDepth < 0 ifTrue:
  [destDepth := 0 - destDepth].
  "Ignore an integer bits handle for Display in which case
  the appropriate values will be obtained by calling ioLockSurfaceBits()."
  (interpreterProxy isIntegerObject: destBits)
  ifTrue: "Query for actual surface dimensions"
  [querySurfaceFn = 0 ifTrue: [self loadSurfacePlugin ifFalse: [^false]].
+ (self querySurfaceFn: (interpreterProxy integerValueOf: destBits)
+ _: (self addressOf: destWidth put: [:v| destWidth := v])
+ _: (self addressOf: destHeight put: [:v| destHeight := v])
+ _: (self addressOf: destDepth put: [:v| destDepth := v])
+ _: (self addressOf: destMSB put: [:v| destMSB := v])) ifFalse:
- (self query: (interpreterProxy integerValueOf: destBits)
- Sur: (self addressOf: destWidth)
- fa: (self addressOf: destHeight)
- ce: (self addressOf: destDepth)
- Fn: (self addressOf: destMSB)) ifFalse:
  [interpreterProxy primitiveFailFor: PrimErrCallbackError.
  ^false].
  destPPW := 32 // destDepth.
  destBits := destPitch := 0]
  ifFalse:
  [(interpreterProxy isWordsOrBytes: destBits) ifFalse:
  [^false].
  destPPW := 32 // destDepth.
  destPitch := destWidth + (destPPW-1) // destPPW * 4.
  destBitsSize := interpreterProxy byteSizeOf: destBits.
  destBitsSize >= (destPitch * destHeight) ifFalse:
  [^false].
  "Skip header since external bits don't have one"
  destBits := self oopForPointer: (interpreterProxy firstIndexableField: destBits)].
  ^true!

Item was changed:
  ----- Method: BitBltSimulation>>loadBitBltSourceForm (in category 'interpreter interface') -----
  loadBitBltSourceForm
  "Load the source form for BitBlt. Return false if anything is wrong, true otherwise."
  | sourceBitsSize |
  <inline: true>
  ((interpreterProxy isPointers: sourceForm)
  and: [(interpreterProxy slotSizeOf: sourceForm) >= 4]) ifFalse:
  [^false].
  sourceBits := interpreterProxy fetchPointer: FormBitsIndex ofObject: sourceForm.
  sourceWidth := self fetchIntOrFloat: FormWidthIndex ofObject: sourceForm.
  sourceHeight := self fetchIntOrFloat: FormHeightIndex ofObject: sourceForm.
  (sourceWidth >= 0 and: [sourceHeight >= 0]) ifFalse:
  [^false].
  sourceDepth := interpreterProxy fetchInteger: FormDepthIndex ofObject: sourceForm.
+ (sourceMSB := sourceDepth > 0) ifFalse:
- sourceMSB := sourceDepth > 0.
- sourceDepth < 0 ifTrue:
  [sourceDepth := 0 - sourceDepth].
  "Ignore an integer bits handle for Display in which case
  the appropriate values will be obtained by calling ioLockSurfaceBits()."
  (interpreterProxy isIntegerObject: sourceBits)
  ifTrue: "Query for actual surface dimensions"
  [querySurfaceFn = 0 ifTrue: [self loadSurfacePlugin ifFalse:[^false]].
+ (self querySurfaceFn: (interpreterProxy integerValueOf: sourceBits)
+ _: (self addressOf: sourceWidth put: [:v| sourceWidth := v])
+ _: (self addressOf: sourceHeight put: [:v| sourceHeight := v])
+ _: (self addressOf: sourceDepth put: [:v| sourceDepth := v])
+ _: (self addressOf: sourceMSB put: [:v| sourceMSB := v])) ifFalse:
- (self query: (interpreterProxy integerValueOf: sourceBits)
- Sur: (self addressOf: sourceWidth)
- fa: (self addressOf: sourceHeight)
- ce: (self addressOf: sourceDepth)
- Fn: (self addressOf: sourceMSB)) ifFalse:
  [interpreterProxy primitiveFailFor: PrimErrCallbackError.
  ^false].
  sourcePPW := 32 // sourceDepth.
  sourceBits := sourcePitch := 0]
  ifFalse:
  [(interpreterProxy isWordsOrBytes: sourceBits) ifFalse:
  [^false].
  sourcePPW := 32 // sourceDepth.
  sourcePitch := sourceWidth + (sourcePPW-1) // sourcePPW * 4.
  sourceBitsSize := interpreterProxy byteSizeOf: sourceBits.
  sourceBitsSize >= (sourcePitch * sourceHeight) ifFalse:
  [^false].
  "Skip header since external bits don't have one"
  sourceBits := self oopForPointer: (interpreterProxy firstIndexableField: sourceBits)].
  ^true!

Item was removed:
- ----- Method: BitBltSimulation>>lock:Su:rf:ac:eF:n: (in category 'surface support') -----
- lock: sourceHandle Su: pitchPtr rf: x ac: y eF: w n: h
- "Simulate the lockSurfaceFn function call as a failure to load the surface."
- <doNotGenerate>
- ^0!

Item was added:
+ ----- Method: BitBltSimulation>>lockSurfaceFn:_:_:_:_:_: (in category 'surface support') -----
+ lockSurfaceFn: sourceHandle _: pitchPtr _: x _: y _: w _: h
+ "Simulate the lockSurfaceFn function call as a failure to load the surface."
+ <doNotGenerate>
+ ^0!

Item was changed:
  ----- Method: BitBltSimulation>>lockSurfaces (in category 'surface support') -----
  lockSurfaces
  "Get a pointer to the bits of any OS surfaces."
  "Notes:
  * For equal source/dest handles only one locking operation is performed.
  This is to prevent locking of overlapping areas which does not work with
  certain APIs (as an example, DirectDraw prevents locking of overlapping areas).
  A special case for non-overlapping but equal source/dest handle would
  be possible but we would have to transfer this information over to
  unlockSurfaces somehow (currently, only one unlock operation is
  performed for equal source and dest handles). Also, this would require
  a change in the notion of ioLockSurface() which is right now interpreted
  as a hint and not as a requirement to lock only the specific portion of
  the surface.
 
  * The arguments in ioLockSurface() provide the implementation with
  an explicit hint what area is affected. It can be very useful to
  know the max. affected area beforehand if getting the bits requires expensive
  copy operations (e.g., like a roundtrip to the X server or a glReadPixel op).
  However, the returned pointer *MUST* point to the virtual origin of the surface
  and not to the beginning of the rectangle. The promise made by BitBlt
  is to never access data outside the given rectangle (aligned to 4byte boundaries!!)
  so it is okay to return a pointer to the virtual origin that is actually outside
  the valid memory area.
 
  * The area provided in ioLockSurface() is already clipped (e.g., it will always
  be inside the source and dest boundingBox) but it is not aligned to word boundaries
  yet. It is up to the support code to compute accurate alignment if necessary.
 
  * Warping always requires the entire source surface to be locked because
  there is no beforehand knowledge about what area will actually be traversed.
 
  * Fail if a GC has occurred since the primitive started (presumably in the lockSurface
    function), because one or more of the primitives' parameters may have been moved.
  "
  | sourceHandle destHandle l r t b |
  <inline: true>
  self assert: numGCsOnInvocation = interpreterProxy statNumGCs.
  hasSurfaceLock := false.
  destBits = 0 ifTrue: "Blitting *to* OS surface"
  [lockSurfaceFn = 0 ifTrue: [self loadSurfacePlugin ifFalse: [^false]].
  destHandle := interpreterProxy fetchInteger: FormBitsIndex ofObject: destForm.
  (sourceBits ~= 0 or: [noSource]) ifFalse:
  [sourceHandle := interpreterProxy fetchInteger: FormBitsIndex ofObject: sourceForm.
  "Handle the special case of equal source and dest handles"
  sourceHandle = destHandle ifTrue:
  "If we have overlapping source/dest we lock the entire area
  so that there is only one area transmitted"
  [isWarping
  ifFalse: "When warping we always need the entire surface for the source"
+ [sourceBits := self lockSurfaceFn: sourceHandle
+ _: (self addressOf: sourcePitch)
+ _: 0
+ _: 0
+ _: sourceWidth
+ _: sourceHeight]
- [sourceBits := self lock: sourceHandle
- Su: (self addressOf: sourcePitch)
- rf: 0
- ac: 0
- eF: sourceWidth
- n: sourceHeight]
  ifTrue: "Otherwise use overlapping area"
  [l := sx min: dx. r := (sx max: dx) + bbW.
+ t := sy min: dy. b := (sy max: dy) + bbH.
+ sourceBits := self lockSurfaceFn: sourceHandle
+ _: (self addressOf: sourcePitch)
+ _: l
+ _: t
+ _: r - l
+ _: b - t].
- t := sy min: dy. b := (sy max: dy) + bbH.
- sourceBits := self lock: sourceHandle
- Su: (self addressOf: sourcePitch)
- rf: l
- ac: t
- eF: r - l
- n: b - t].
  destBits := sourceBits.
  destPitch := sourcePitch.
  hasSurfaceLock := true.
  numGCsOnInvocation ~= interpreterProxy statNumGCs ifTrue:
  [self unlockSurfaces.
  interpreterProxy primitiveFailFor: PrimErrObjectMoved.
  ^false].
  destBits = 0 ifTrue:
  [self unlockSurfaces.
  interpreterProxy primitiveFailFor: PrimErrCallbackError.
  ^false].
+ endOfDestination := endOfSource := sourceBits + (sourcePitch * sourceHeight).
  ^true]].
  "Fall through - if not equal it'll be handled below"
+ destBits := self lockSurfaceFn: destHandle
+ _: (self addressOf: destPitch)
+ _: dx
+ _: dy
+ _: bbW
+ _: bbH.
- destBits := self lock: destHandle
- Su: (self addressOf: destPitch)
- rf: dx
- ac: dy
- eF: bbW
- n: bbH.
  hasSurfaceLock := true.
  numGCsOnInvocation ~= interpreterProxy statNumGCs ifTrue:
  [self unlockSurfaces.
  interpreterProxy primitiveFailFor: PrimErrObjectMoved.
  ^false].
  destBits = 0 ifTrue:
  [interpreterProxy primitiveFailFor: PrimErrCallbackError]].
 
  (sourceBits ~= 0 or: [noSource]) ifFalse: "Blitting *from* OS surface"
  [sourceHandle := interpreterProxy fetchInteger: FormBitsIndex ofObject: sourceForm.
  interpreterProxy failed ifTrue: [^false]. "fetch sourceHandle could fail"
  lockSurfaceFn = 0 ifTrue:[self loadSurfacePlugin ifFalse:[^false]].
  "Warping requiring the entire surface"
  isWarping
  ifTrue:
+ [sourceBits := self lockSurfaceFn: sourceHandle
+ _: (self addressOf: sourcePitch)
+ _: 0
+ _: 0
+ _: sourceWidth
+ _: sourceHeight]
- [sourceBits := self lock: sourceHandle
- Su: (self addressOf: sourcePitch)
- rf: 0
- ac: 0
- eF: sourceWidth
- n: sourceHeight]
  ifFalse:
+ [sourceBits := self lockSurfaceFn: sourceHandle
+ _: (self addressOf: sourcePitch)
+ _: sx
+ _: sy
+ _: bbW
+ _: bbH].
- [sourceBits := self lock: sourceHandle
- Su: (self addressOf: sourcePitch)
- rf: sx
- ac: sy
- eF: bbW
- n: bbH].
  hasSurfaceLock := true.
  numGCsOnInvocation ~= interpreterProxy statNumGCs ifTrue:
  [self unlockSurfaces.
  interpreterProxy primitiveFailFor: PrimErrObjectMoved.
  ^false].
  sourceBits = 0 ifTrue:
  [interpreterProxy primitiveFailFor: PrimErrCallbackError]].
+ endOfSource := sourceBits + (sourcePitch * sourceHeight).
+ endOfDestination := destBits + (destPitch * destHeight).
  ^destBits ~= 0 and: [sourceBits ~= 0 or: [noSource]]!

Item was removed:
- ----- Method: BitBltSimulation>>query:Sur:fa:ce:Fn: (in category 'surface support') -----
- query: handle Sur: widthPtr fa: heightPtr ce: depthPtr Fn: endianPtr
- "Query the dimension of an OS surface.
- This method is provided so that in case the inst vars of the
- source form are broken, *actual* values of the OS surface
- can be obtained. This might, for instance, happen if the user
- resizes the main window.
- This is a simulation of the querySurfaceFn function call; simulate as a failure."
- <doNotGenerate>
- ^false!

Item was added:
+ ----- Method: BitBltSimulation>>querySurfaceFn:_:_:_:_: (in category 'surface support') -----
+ querySurfaceFn: handle _: widthPtr _: heightPtr _: depthPtr _: endianPtr
+ "Query the dimension of an OS surface.
+ This method is provided so that in case the inst vars of the
+ source form are broken, *actual* values of the OS surface
+ can be obtained. This might, for instance, happen if the user
+ resizes the main window.
+ This is a simulation of the querySurfaceFn function call; simulate as a failure."
+ <doNotGenerate>
+ ^false!

Item was changed:
  ----- Method: BitBltSimulation>>srcLongAt: (in category 'memory access') -----
  srcLongAt: idx
+ <inline: #always>
+ self assert: idx asUnsignedInteger < endOfSource.
-
  ^self long32At: idx!

Item was changed:
  ----- Method: BitBltSimulation>>tallyMapAt: (in category 'memory access') -----
  tallyMapAt: idx
  "Return the word at position idx from the colorMap"
+ <inline: #always>
  ^cmLookupTable at: (idx bitAnd: cmMask)!

Item was changed:
  ----- Method: BitBltSimulation>>tallyMapAt:put: (in category 'memory access') -----
  tallyMapAt: idx put: value
  "Store the word at position idx in the colorMap"
+ <inline: #always>
  ^cmLookupTable at: (idx bitAnd: cmMask) put: value!

Item was removed:
- ----- Method: BitBltSimulation>>unlock:Sur:fa:ce:Fn: (in category 'surface support') -----
- unlock: handle Sur: x fa: y ce: w Fn: h
- "Simulate the unlockSurfaceFn function call."
- <doNotGenerate>
- ^self!

Item was changed:
  ----- Method: BitBltSimulation>>unlockSurfaces (in category 'surface support') -----
  unlockSurfaces
  "Unlock the bits of any OS surfaces."
  "See the comment in lockSurfaces. Similar rules apply. That is, the area provided in ioUnlockSurface
  can be used to determine the dirty region after drawing. If a source is unlocked, then the area will
  be (0,0,0,0) to indicate that no portion is dirty.  Note that if a GC happens during unlockSourceFn
  (if it is effectively a callback) no matter.  No bits are touched after unlock."
  | sourceHandle destHandle destLocked |
  hasSurfaceLock ifFalse: [^self].
  unlockSurfaceFn = 0 ifTrue:
  [self loadSurfacePlugin ifFalse:
  [^self]].
  self ensureDestAndSourceFormsAreValid.
  destLocked := false.
  destHandle := interpreterProxy fetchPointer: FormBitsIndex ofObject: destForm.
  (interpreterProxy isIntegerObject: destHandle) ifTrue: "The destBits are always assumed to be dirty"
+ [self unlockSurfaceFn: (interpreterProxy integerValueOf: destHandle)
+ _: affectedL
+ _: affectedT
+ _: affectedR - affectedL
+ _: affectedB - affectedT.
- [self unlock: (interpreterProxy integerValueOf: destHandle)
- Sur: affectedL
- fa: affectedT
- ce: affectedR - affectedL
- Fn: affectedB - affectedT.
  destBits := destPitch := 0.
  destLocked := true].
  noSource ifFalse:
  [self ensureDestAndSourceFormsAreValid.
  sourceHandle := interpreterProxy fetchPointer: FormBitsIndex ofObject: sourceForm.
  (interpreterProxy isIntegerObject: sourceHandle) ifTrue:
  ["Only unlock sourceHandle if different from destHandle"
  (destLocked and: [sourceHandle = destHandle]) ifFalse:
  [self unlock: (interpreterProxy integerValueOf: sourceHandle) Sur: 0 fa: 0 ce: 0 Fn: 0].
  sourceBits := sourcePitch := 0]].
  hasSurfaceLock := false.
  self cCode: [] inSmalltalk:
  [interpreterProxy displayObject = destForm ifTrue:
  [interpreterProxy getDeferDisplayUpdates ifFalse:
  [interpreterProxy fullDisplayUpdate]]]!

Item was changed:
  ----- Method: BitBltSimulator>>dstLongAt: (in category 'debug support') -----
  dstLongAt: dstIndex
+ interpreterProxy isInterpreterProxy ifTrue:
+ [^dstIndex long32At: 0].
+ ^super dstLongAt: dstIndex!
-
- interpreterProxy isInterpreterProxy
- ifTrue:[^dstIndex long32At: 0].
- ((dstIndex anyMask: 3) or:[dstIndex + 4 < destBits or:[
- dstIndex > (destBits + (destPitch * destHeight))]])
- ifTrue:[self error:'Out of bounds'].
- ^self long32At: dstIndex!

Item was changed:
  ----- Method: BitBltSimulator>>dstLongAt:put: (in category 'debug support') -----
  dstLongAt: dstIndex put: value
+ interpreterProxy isInterpreterProxy ifTrue:
+ [^dstIndex long32At: 0 put: value].
+ ^super dstLongAt: dstIndex put: value!
-
- interpreterProxy isInterpreterProxy
- ifTrue:[^dstIndex long32At: 0 put: value].
- ((dstIndex anyMask: 3) or:[dstIndex < destBits or:[
- dstIndex >= (destBits + (destPitch * destHeight))]])
- ifTrue:[self error:'Out of bounds'].
- ^self long32At: dstIndex put: value!

Item was changed:
  ----- Method: BitBltSimulator>>halftoneAt: (in category 'memory access') -----
  halftoneAt: idx
+ interpreterProxy isInterpreterProxy ifTrue:
+ [^(halftoneBase + (idx \\ halftoneHeight * 4)) long32At: 0].
+ ^super halftoneAt: idx!
-
- ^self
- cCode: [(halftoneBase + (idx \\ halftoneHeight * 4)) long32At: 0]
- inSmalltalk: [super halftoneAt: idx]!

Item was changed:
  ----- Method: BitBltSimulator>>srcLongAt: (in category 'debug support') -----
  srcLongAt: srcIndex
+ interpreterProxy isInterpreterProxy ifTrue:
+ [^srcIndex long32At: 0].
+ ^super srcLongAt: srcIndex!
-
- interpreterProxy isInterpreterProxy
- ifTrue:[^srcIndex long32At: 0].
- ((srcIndex anyMask: 3) or:[srcIndex + 4 < sourceBits or:[
- srcIndex > (sourceBits + (sourcePitch * sourceHeight))]])
- ifTrue:[self error:'Out of bounds'].
- ^self long32At: srcIndex!