| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2011 The Android Open Source Project | 
|  | 3 | * | 
|  | 4 | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 5 | * you may not use this file except in compliance with the License. | 
|  | 6 | * You may obtain a copy of the License at | 
|  | 7 | * | 
|  | 8 | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 9 | * | 
|  | 10 | * Unless required by applicable law or agreed to in writing, software | 
|  | 11 | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 13 | * See the License for the specific language governing permissions and | 
|  | 14 | * limitations under the License. | 
|  | 15 | */ | 
|  | 16 |  | 
|  | 17 | #define LOG_TAG "Sprites" | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 18 | //#define LOG_NDEBUG 0 | 
|  | 19 |  | 
|  | 20 | #include "SpriteController.h" | 
|  | 21 |  | 
| Mark Salyzyn | 52eb4e0 | 2016-09-28 16:15:30 -0700 | [diff] [blame] | 22 | #include <log/log.h> | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 23 | #include <utils/String8.h> | 
| Mathias Agopian | 5280061 | 2013-02-14 17:11:20 -0800 | [diff] [blame] | 24 | #include <gui/Surface.h> | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 25 |  | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 26 | namespace android { | 
|  | 27 |  | 
|  | 28 | // --- SpriteController --- | 
|  | 29 |  | 
| Prabir Pradhan | fef0c61 | 2021-11-04 14:08:50 -0700 | [diff] [blame] | 30 | SpriteController::SpriteController(const sp<Looper>& looper, int32_t overlayLayer, | 
|  | 31 | ParentSurfaceProvider parentSurfaceProvider) | 
|  | 32 | : mLooper(looper), | 
|  | 33 | mOverlayLayer(overlayLayer), | 
|  | 34 | mParentSurfaceProvider(std::move(parentSurfaceProvider)) { | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 35 | mHandler = new WeakMessageHandler(this); | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 36 | mLocked.transactionNestingCount = 0; | 
|  | 37 | mLocked.deferredSpriteUpdate = false; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 38 | } | 
|  | 39 |  | 
|  | 40 | SpriteController::~SpriteController() { | 
|  | 41 | mLooper->removeMessages(mHandler); | 
|  | 42 |  | 
|  | 43 | if (mSurfaceComposerClient != NULL) { | 
|  | 44 | mSurfaceComposerClient->dispose(); | 
|  | 45 | mSurfaceComposerClient.clear(); | 
|  | 46 | } | 
|  | 47 | } | 
|  | 48 |  | 
|  | 49 | sp<Sprite> SpriteController::createSprite() { | 
|  | 50 | return new SpriteImpl(this); | 
|  | 51 | } | 
|  | 52 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 53 | void SpriteController::openTransaction() { | 
|  | 54 | AutoMutex _l(mLock); | 
|  | 55 |  | 
|  | 56 | mLocked.transactionNestingCount += 1; | 
|  | 57 | } | 
|  | 58 |  | 
|  | 59 | void SpriteController::closeTransaction() { | 
|  | 60 | AutoMutex _l(mLock); | 
|  | 61 |  | 
|  | 62 | LOG_ALWAYS_FATAL_IF(mLocked.transactionNestingCount == 0, | 
|  | 63 | "Sprite closeTransaction() called but there is no open sprite transaction"); | 
|  | 64 |  | 
|  | 65 | mLocked.transactionNestingCount -= 1; | 
|  | 66 | if (mLocked.transactionNestingCount == 0 && mLocked.deferredSpriteUpdate) { | 
|  | 67 | mLocked.deferredSpriteUpdate = false; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 68 | mLooper->sendMessage(mHandler, Message(MSG_UPDATE_SPRITES)); | 
|  | 69 | } | 
|  | 70 | } | 
|  | 71 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 72 | void SpriteController::invalidateSpriteLocked(const sp<SpriteImpl>& sprite) { | 
| Prabir Pradhan | 3f8b289 | 2021-11-18 08:55:02 -0800 | [diff] [blame] | 73 | bool wasEmpty = mLocked.invalidatedSprites.empty(); | 
|  | 74 | mLocked.invalidatedSprites.push_back(sprite); | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 75 | if (wasEmpty) { | 
|  | 76 | if (mLocked.transactionNestingCount != 0) { | 
|  | 77 | mLocked.deferredSpriteUpdate = true; | 
|  | 78 | } else { | 
|  | 79 | mLooper->sendMessage(mHandler, Message(MSG_UPDATE_SPRITES)); | 
|  | 80 | } | 
|  | 81 | } | 
|  | 82 | } | 
|  | 83 |  | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 84 | void SpriteController::disposeSurfaceLocked(const sp<SurfaceControl>& surfaceControl) { | 
| Prabir Pradhan | 3f8b289 | 2021-11-18 08:55:02 -0800 | [diff] [blame] | 85 | bool wasEmpty = mLocked.disposedSurfaces.empty(); | 
|  | 86 | mLocked.disposedSurfaces.push_back(surfaceControl); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 87 | if (wasEmpty) { | 
|  | 88 | mLooper->sendMessage(mHandler, Message(MSG_DISPOSE_SURFACES)); | 
|  | 89 | } | 
|  | 90 | } | 
|  | 91 |  | 
|  | 92 | void SpriteController::handleMessage(const Message& message) { | 
|  | 93 | switch (message.what) { | 
|  | 94 | case MSG_UPDATE_SPRITES: | 
|  | 95 | doUpdateSprites(); | 
|  | 96 | break; | 
|  | 97 | case MSG_DISPOSE_SURFACES: | 
|  | 98 | doDisposeSurfaces(); | 
|  | 99 | break; | 
|  | 100 | } | 
|  | 101 | } | 
|  | 102 |  | 
|  | 103 | void SpriteController::doUpdateSprites() { | 
|  | 104 | // Collect information about sprite updates. | 
|  | 105 | // Each sprite update record includes a reference to its associated sprite so we can | 
|  | 106 | // be certain the sprites will not be deleted while this function runs.  Sprites | 
|  | 107 | // may invalidate themselves again during this time but we will handle those changes | 
|  | 108 | // in the next iteration. | 
|  | 109 | Vector<SpriteUpdate> updates; | 
|  | 110 | size_t numSprites; | 
|  | 111 | { // acquire lock | 
|  | 112 | AutoMutex _l(mLock); | 
|  | 113 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 114 | numSprites = mLocked.invalidatedSprites.size(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 115 | for (size_t i = 0; i < numSprites; i++) { | 
| Prabir Pradhan | 3f8b289 | 2021-11-18 08:55:02 -0800 | [diff] [blame] | 116 | const sp<SpriteImpl>& sprite = mLocked.invalidatedSprites[i]; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 117 |  | 
|  | 118 | updates.push(SpriteUpdate(sprite, sprite->getStateLocked())); | 
|  | 119 | sprite->resetDirtyLocked(); | 
|  | 120 | } | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 121 | mLocked.invalidatedSprites.clear(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 122 | } // release lock | 
|  | 123 |  | 
|  | 124 | // Create missing surfaces. | 
|  | 125 | bool surfaceChanged = false; | 
|  | 126 | for (size_t i = 0; i < numSprites; i++) { | 
|  | 127 | SpriteUpdate& update = updates.editItemAt(i); | 
|  | 128 |  | 
|  | 129 | if (update.state.surfaceControl == NULL && update.state.wantSurfaceVisible()) { | 
| Garfield Tan | 7e3457e | 2020-05-28 14:31:29 -0700 | [diff] [blame] | 130 | update.state.surfaceWidth = update.state.icon.width(); | 
|  | 131 | update.state.surfaceHeight = update.state.icon.height(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 132 | update.state.surfaceDrawn = false; | 
|  | 133 | update.state.surfaceVisible = false; | 
|  | 134 | update.state.surfaceControl = obtainSurface( | 
|  | 135 | update.state.surfaceWidth, update.state.surfaceHeight); | 
|  | 136 | if (update.state.surfaceControl != NULL) { | 
|  | 137 | update.surfaceChanged = surfaceChanged = true; | 
|  | 138 | } | 
|  | 139 | } | 
|  | 140 | } | 
|  | 141 |  | 
| Arthur Hung | b9b3200 | 2018-12-18 17:39:43 +0800 | [diff] [blame] | 142 | // Resize and/or reparent sprites if needed. | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 143 | SurfaceComposerClient::Transaction t; | 
|  | 144 | bool needApplyTransaction = false; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 145 | for (size_t i = 0; i < numSprites; i++) { | 
|  | 146 | SpriteUpdate& update = updates.editItemAt(i); | 
| Arthur Hung | b9b3200 | 2018-12-18 17:39:43 +0800 | [diff] [blame] | 147 | if (update.state.surfaceControl == nullptr) { | 
|  | 148 | continue; | 
|  | 149 | } | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 150 |  | 
| Arthur Hung | b9b3200 | 2018-12-18 17:39:43 +0800 | [diff] [blame] | 151 | if (update.state.wantSurfaceVisible()) { | 
| Garfield Tan | 7e3457e | 2020-05-28 14:31:29 -0700 | [diff] [blame] | 152 | int32_t desiredWidth = update.state.icon.width(); | 
|  | 153 | int32_t desiredHeight = update.state.icon.height(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 154 | if (update.state.surfaceWidth < desiredWidth | 
|  | 155 | || update.state.surfaceHeight < desiredHeight) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 156 | needApplyTransaction = true; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 157 |  | 
| Prabir Pradhan | 6272659 | 2021-08-09 15:51:25 +0000 | [diff] [blame] | 158 | update.state.surfaceControl->updateDefaultBufferSize(desiredWidth, desiredHeight); | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 159 | update.state.surfaceWidth = desiredWidth; | 
|  | 160 | update.state.surfaceHeight = desiredHeight; | 
|  | 161 | update.state.surfaceDrawn = false; | 
|  | 162 | update.surfaceChanged = surfaceChanged = true; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 163 |  | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 164 | if (update.state.surfaceVisible) { | 
|  | 165 | t.hide(update.state.surfaceControl); | 
|  | 166 | update.state.surfaceVisible = false; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 167 | } | 
|  | 168 | } | 
|  | 169 | } | 
| Arthur Hung | b9b3200 | 2018-12-18 17:39:43 +0800 | [diff] [blame] | 170 |  | 
|  | 171 | // If surface is a new one, we have to set right layer stack. | 
|  | 172 | if (update.surfaceChanged || update.state.dirty & DIRTY_DISPLAY_ID) { | 
| Prabir Pradhan | fef0c61 | 2021-11-04 14:08:50 -0700 | [diff] [blame] | 173 | t.reparent(update.state.surfaceControl, mParentSurfaceProvider(update.state.displayId)); | 
| Arthur Hung | b9b3200 | 2018-12-18 17:39:43 +0800 | [diff] [blame] | 174 | needApplyTransaction = true; | 
|  | 175 | } | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 176 | } | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 177 | if (needApplyTransaction) { | 
|  | 178 | t.apply(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 179 | } | 
|  | 180 |  | 
|  | 181 | // Redraw sprites if needed. | 
|  | 182 | for (size_t i = 0; i < numSprites; i++) { | 
|  | 183 | SpriteUpdate& update = updates.editItemAt(i); | 
|  | 184 |  | 
|  | 185 | if ((update.state.dirty & DIRTY_BITMAP) && update.state.surfaceDrawn) { | 
|  | 186 | update.state.surfaceDrawn = false; | 
|  | 187 | update.surfaceChanged = surfaceChanged = true; | 
|  | 188 | } | 
|  | 189 |  | 
|  | 190 | if (update.state.surfaceControl != NULL && !update.state.surfaceDrawn | 
|  | 191 | && update.state.wantSurfaceVisible()) { | 
|  | 192 | sp<Surface> surface = update.state.surfaceControl->getSurface(); | 
| Garfield Tan | 7e3457e | 2020-05-28 14:31:29 -0700 | [diff] [blame] | 193 | if (update.state.icon.draw(surface)) { | 
|  | 194 | update.state.surfaceDrawn = true; | 
|  | 195 | update.surfaceChanged = surfaceChanged = true; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 196 | } | 
|  | 197 | } | 
|  | 198 | } | 
|  | 199 |  | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 200 | needApplyTransaction = false; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 201 | for (size_t i = 0; i < numSprites; i++) { | 
|  | 202 | SpriteUpdate& update = updates.editItemAt(i); | 
|  | 203 |  | 
|  | 204 | bool wantSurfaceVisibleAndDrawn = update.state.wantSurfaceVisible() | 
|  | 205 | && update.state.surfaceDrawn; | 
|  | 206 | bool becomingVisible = wantSurfaceVisibleAndDrawn && !update.state.surfaceVisible; | 
|  | 207 | bool becomingHidden = !wantSurfaceVisibleAndDrawn && update.state.surfaceVisible; | 
|  | 208 | if (update.state.surfaceControl != NULL && (becomingVisible || becomingHidden | 
|  | 209 | || (wantSurfaceVisibleAndDrawn && (update.state.dirty & (DIRTY_ALPHA | 
|  | 210 | | DIRTY_POSITION | DIRTY_TRANSFORMATION_MATRIX | DIRTY_LAYER | 
| Garfield Tan | 67e479a | 2019-08-05 16:47:40 -0700 | [diff] [blame] | 211 | | DIRTY_VISIBILITY | DIRTY_HOTSPOT | DIRTY_DISPLAY_ID | 
|  | 212 | | DIRTY_ICON_STYLE))))) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 213 | needApplyTransaction = true; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 214 |  | 
|  | 215 | if (wantSurfaceVisibleAndDrawn | 
|  | 216 | && (becomingVisible || (update.state.dirty & DIRTY_ALPHA))) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 217 | t.setAlpha(update.state.surfaceControl, | 
|  | 218 | update.state.alpha); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 219 | } | 
|  | 220 |  | 
|  | 221 | if (wantSurfaceVisibleAndDrawn | 
|  | 222 | && (becomingVisible || (update.state.dirty & (DIRTY_POSITION | 
|  | 223 | | DIRTY_HOTSPOT)))) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 224 | t.setPosition( | 
|  | 225 | update.state.surfaceControl, | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 226 | update.state.positionX - update.state.icon.hotSpotX, | 
|  | 227 | update.state.positionY - update.state.icon.hotSpotY); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 228 | } | 
|  | 229 |  | 
|  | 230 | if (wantSurfaceVisibleAndDrawn | 
|  | 231 | && (becomingVisible | 
|  | 232 | || (update.state.dirty & DIRTY_TRANSFORMATION_MATRIX))) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 233 | t.setMatrix( | 
|  | 234 | update.state.surfaceControl, | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 235 | update.state.transformationMatrix.dsdx, | 
|  | 236 | update.state.transformationMatrix.dtdx, | 
|  | 237 | update.state.transformationMatrix.dsdy, | 
|  | 238 | update.state.transformationMatrix.dtdy); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 239 | } | 
|  | 240 |  | 
| Garfield Tan | 67e479a | 2019-08-05 16:47:40 -0700 | [diff] [blame] | 241 | if (wantSurfaceVisibleAndDrawn | 
|  | 242 | && (becomingVisible | 
|  | 243 | || (update.state.dirty & (DIRTY_HOTSPOT | DIRTY_ICON_STYLE)))) { | 
|  | 244 | Parcel p; | 
|  | 245 | p.writeInt32(update.state.icon.style); | 
|  | 246 | p.writeFloat(update.state.icon.hotSpotX); | 
|  | 247 | p.writeFloat(update.state.icon.hotSpotY); | 
|  | 248 |  | 
|  | 249 | // Pass cursor metadata in the sprite surface so that when Android is running as a | 
|  | 250 | // client OS (e.g. ARC++) the host OS can get the requested cursor metadata and | 
|  | 251 | // update mouse cursor in the host OS. | 
| Huihong Luo | 36b55bc | 2022-03-08 14:50:45 -0800 | [diff] [blame] | 252 | t.setMetadata(update.state.surfaceControl, gui::METADATA_MOUSE_CURSOR, p); | 
| Garfield Tan | 67e479a | 2019-08-05 16:47:40 -0700 | [diff] [blame] | 253 | } | 
|  | 254 |  | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 255 | int32_t surfaceLayer = mOverlayLayer + update.state.layer; | 
|  | 256 | if (wantSurfaceVisibleAndDrawn | 
|  | 257 | && (becomingVisible || (update.state.dirty & DIRTY_LAYER))) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 258 | t.setLayer(update.state.surfaceControl, surfaceLayer); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 259 | } | 
|  | 260 |  | 
|  | 261 | if (becomingVisible) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 262 | t.show(update.state.surfaceControl); | 
|  | 263 |  | 
|  | 264 | update.state.surfaceVisible = true; | 
|  | 265 | update.surfaceChanged = surfaceChanged = true; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 266 | } else if (becomingHidden) { | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 267 | t.hide(update.state.surfaceControl); | 
|  | 268 |  | 
|  | 269 | update.state.surfaceVisible = false; | 
|  | 270 | update.surfaceChanged = surfaceChanged = true; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 271 | } | 
|  | 272 | } | 
|  | 273 | } | 
|  | 274 |  | 
| Robert Carr | e13b58e | 2017-08-31 14:50:44 -0700 | [diff] [blame] | 275 | if (needApplyTransaction) { | 
|  | 276 | status_t status = t.apply(); | 
|  | 277 | if (status) { | 
|  | 278 | ALOGE("Error applying Surface transaction"); | 
|  | 279 | } | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 280 | } | 
|  | 281 |  | 
|  | 282 | // If any surfaces were changed, write back the new surface properties to the sprites. | 
|  | 283 | if (surfaceChanged) { // acquire lock | 
|  | 284 | AutoMutex _l(mLock); | 
|  | 285 |  | 
|  | 286 | for (size_t i = 0; i < numSprites; i++) { | 
|  | 287 | const SpriteUpdate& update = updates.itemAt(i); | 
|  | 288 |  | 
|  | 289 | if (update.surfaceChanged) { | 
|  | 290 | update.sprite->setSurfaceLocked(update.state.surfaceControl, | 
|  | 291 | update.state.surfaceWidth, update.state.surfaceHeight, | 
|  | 292 | update.state.surfaceDrawn, update.state.surfaceVisible); | 
|  | 293 | } | 
|  | 294 | } | 
|  | 295 | } // release lock | 
|  | 296 |  | 
|  | 297 | // Clear the sprite update vector outside the lock.  It is very important that | 
|  | 298 | // we do not clear sprite references inside the lock since we could be releasing | 
|  | 299 | // the last remaining reference to the sprite here which would result in the | 
|  | 300 | // sprite being deleted and the lock being reacquired by the sprite destructor | 
|  | 301 | // while already held. | 
|  | 302 | updates.clear(); | 
|  | 303 | } | 
|  | 304 |  | 
|  | 305 | void SpriteController::doDisposeSurfaces() { | 
|  | 306 | // Collect disposed surfaces. | 
| Prabir Pradhan | 3f8b289 | 2021-11-18 08:55:02 -0800 | [diff] [blame] | 307 | std::vector<sp<SurfaceControl>> disposedSurfaces; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 308 | { // acquire lock | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 309 | AutoMutex _l(mLock); | 
|  | 310 |  | 
|  | 311 | disposedSurfaces = mLocked.disposedSurfaces; | 
|  | 312 | mLocked.disposedSurfaces.clear(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 313 | } // release lock | 
|  | 314 |  | 
| Prabir Pradhan | 3f8b289 | 2021-11-18 08:55:02 -0800 | [diff] [blame] | 315 | // Remove the parent from all surfaces. | 
|  | 316 | SurfaceComposerClient::Transaction t; | 
|  | 317 | for (const sp<SurfaceControl>& sc : disposedSurfaces) { | 
|  | 318 | t.reparent(sc, nullptr); | 
|  | 319 | } | 
|  | 320 | t.apply(); | 
|  | 321 |  | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 322 | // Release the last reference to each surface outside of the lock. | 
|  | 323 | // We don't want the surfaces to be deleted while we are holding our lock. | 
|  | 324 | disposedSurfaces.clear(); | 
|  | 325 | } | 
|  | 326 |  | 
|  | 327 | void SpriteController::ensureSurfaceComposerClient() { | 
|  | 328 | if (mSurfaceComposerClient == NULL) { | 
|  | 329 | mSurfaceComposerClient = new SurfaceComposerClient(); | 
|  | 330 | } | 
|  | 331 | } | 
|  | 332 |  | 
|  | 333 | sp<SurfaceControl> SpriteController::obtainSurface(int32_t width, int32_t height) { | 
|  | 334 | ensureSurfaceComposerClient(); | 
|  | 335 |  | 
|  | 336 | sp<SurfaceControl> surfaceControl = mSurfaceComposerClient->createSurface( | 
| Jeff Brown | 64a55af | 2012-08-26 02:47:39 -0700 | [diff] [blame] | 337 | String8("Sprite"), width, height, PIXEL_FORMAT_RGBA_8888, | 
| Riley Andrews | 68eccda | 2014-07-07 11:47:35 -0700 | [diff] [blame] | 338 | ISurfaceComposerClient::eHidden | | 
|  | 339 | ISurfaceComposerClient::eCursorWindow); | 
| Mathias Agopian | 5280061 | 2013-02-14 17:11:20 -0800 | [diff] [blame] | 340 | if (surfaceControl == NULL || !surfaceControl->isValid()) { | 
| Steve Block | 3762c31 | 2012-01-06 19:20:56 +0000 | [diff] [blame] | 341 | ALOGE("Error creating sprite surface."); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 342 | return NULL; | 
|  | 343 | } | 
|  | 344 | return surfaceControl; | 
|  | 345 | } | 
|  | 346 |  | 
|  | 347 |  | 
|  | 348 | // --- SpriteController::SpriteImpl --- | 
|  | 349 |  | 
|  | 350 | SpriteController::SpriteImpl::SpriteImpl(const sp<SpriteController> controller) : | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 351 | mController(controller) { | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 352 | } | 
|  | 353 |  | 
|  | 354 | SpriteController::SpriteImpl::~SpriteImpl() { | 
|  | 355 | AutoMutex _m(mController->mLock); | 
|  | 356 |  | 
|  | 357 | // Let the controller take care of deleting the last reference to sprite | 
|  | 358 | // surfaces so that we do not block the caller on an IPC here. | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 359 | if (mLocked.state.surfaceControl != NULL) { | 
|  | 360 | mController->disposeSurfaceLocked(mLocked.state.surfaceControl); | 
|  | 361 | mLocked.state.surfaceControl.clear(); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 362 | } | 
|  | 363 | } | 
|  | 364 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 365 | void SpriteController::SpriteImpl::setIcon(const SpriteIcon& icon) { | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 366 | AutoMutex _l(mController->mLock); | 
|  | 367 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 368 | uint32_t dirty; | 
|  | 369 | if (icon.isValid()) { | 
| Derek Sollenberger | 9ca5bbe | 2019-08-14 15:50:59 -0400 | [diff] [blame] | 370 | mLocked.state.icon.bitmap = icon.bitmap.copy(ANDROID_BITMAP_FORMAT_RGBA_8888); | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 371 | if (!mLocked.state.icon.isValid() | 
|  | 372 | || mLocked.state.icon.hotSpotX != icon.hotSpotX | 
|  | 373 | || mLocked.state.icon.hotSpotY != icon.hotSpotY) { | 
|  | 374 | mLocked.state.icon.hotSpotX = icon.hotSpotX; | 
|  | 375 | mLocked.state.icon.hotSpotY = icon.hotSpotY; | 
|  | 376 | dirty = DIRTY_BITMAP | DIRTY_HOTSPOT; | 
|  | 377 | } else { | 
|  | 378 | dirty = DIRTY_BITMAP; | 
|  | 379 | } | 
| Garfield Tan | 67e479a | 2019-08-05 16:47:40 -0700 | [diff] [blame] | 380 |  | 
|  | 381 | if (mLocked.state.icon.style != icon.style) { | 
|  | 382 | mLocked.state.icon.style = icon.style; | 
|  | 383 | dirty |= DIRTY_ICON_STYLE; | 
|  | 384 | } | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 385 | } else if (mLocked.state.icon.isValid()) { | 
|  | 386 | mLocked.state.icon.bitmap.reset(); | 
| Garfield Tan | 67e479a | 2019-08-05 16:47:40 -0700 | [diff] [blame] | 387 | dirty = DIRTY_BITMAP | DIRTY_HOTSPOT | DIRTY_ICON_STYLE; | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 388 | } else { | 
|  | 389 | return; // setting to invalid icon and already invalid so nothing to do | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 390 | } | 
|  | 391 |  | 
|  | 392 | invalidateLocked(dirty); | 
|  | 393 | } | 
|  | 394 |  | 
|  | 395 | void SpriteController::SpriteImpl::setVisible(bool visible) { | 
|  | 396 | AutoMutex _l(mController->mLock); | 
|  | 397 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 398 | if (mLocked.state.visible != visible) { | 
|  | 399 | mLocked.state.visible = visible; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 400 | invalidateLocked(DIRTY_VISIBILITY); | 
|  | 401 | } | 
|  | 402 | } | 
|  | 403 |  | 
|  | 404 | void SpriteController::SpriteImpl::setPosition(float x, float y) { | 
|  | 405 | AutoMutex _l(mController->mLock); | 
|  | 406 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 407 | if (mLocked.state.positionX != x || mLocked.state.positionY != y) { | 
|  | 408 | mLocked.state.positionX = x; | 
|  | 409 | mLocked.state.positionY = y; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 410 | invalidateLocked(DIRTY_POSITION); | 
|  | 411 | } | 
|  | 412 | } | 
|  | 413 |  | 
|  | 414 | void SpriteController::SpriteImpl::setLayer(int32_t layer) { | 
|  | 415 | AutoMutex _l(mController->mLock); | 
|  | 416 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 417 | if (mLocked.state.layer != layer) { | 
|  | 418 | mLocked.state.layer = layer; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 419 | invalidateLocked(DIRTY_LAYER); | 
|  | 420 | } | 
|  | 421 | } | 
|  | 422 |  | 
|  | 423 | void SpriteController::SpriteImpl::setAlpha(float alpha) { | 
|  | 424 | AutoMutex _l(mController->mLock); | 
|  | 425 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 426 | if (mLocked.state.alpha != alpha) { | 
|  | 427 | mLocked.state.alpha = alpha; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 428 | invalidateLocked(DIRTY_ALPHA); | 
|  | 429 | } | 
|  | 430 | } | 
|  | 431 |  | 
|  | 432 | void SpriteController::SpriteImpl::setTransformationMatrix( | 
|  | 433 | const SpriteTransformationMatrix& matrix) { | 
|  | 434 | AutoMutex _l(mController->mLock); | 
|  | 435 |  | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 436 | if (mLocked.state.transformationMatrix != matrix) { | 
|  | 437 | mLocked.state.transformationMatrix = matrix; | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 438 | invalidateLocked(DIRTY_TRANSFORMATION_MATRIX); | 
|  | 439 | } | 
|  | 440 | } | 
|  | 441 |  | 
| Arthur Hung | b9b3200 | 2018-12-18 17:39:43 +0800 | [diff] [blame] | 442 | void SpriteController::SpriteImpl::setDisplayId(int32_t displayId) { | 
|  | 443 | AutoMutex _l(mController->mLock); | 
|  | 444 |  | 
|  | 445 | if (mLocked.state.displayId != displayId) { | 
|  | 446 | mLocked.state.displayId = displayId; | 
|  | 447 | invalidateLocked(DIRTY_DISPLAY_ID); | 
|  | 448 | } | 
|  | 449 | } | 
|  | 450 |  | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 451 | void SpriteController::SpriteImpl::invalidateLocked(uint32_t dirty) { | 
| Jeff Brown | 2352b97 | 2011-04-12 22:39:53 -0700 | [diff] [blame] | 452 | bool wasDirty = mLocked.state.dirty; | 
|  | 453 | mLocked.state.dirty |= dirty; | 
|  | 454 |  | 
|  | 455 | if (!wasDirty) { | 
|  | 456 | mController->invalidateSpriteLocked(this); | 
| Jeff Brown | 5541de9 | 2011-04-11 11:54:25 -0700 | [diff] [blame] | 457 | } | 
|  | 458 | } | 
|  | 459 |  | 
|  | 460 | } // namespace android |