| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2007 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 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 17 | #include <stdlib.h> | 
|  | 18 | #include <stdint.h> | 
|  | 19 | #include <sys/types.h> | 
|  | 20 |  | 
|  | 21 | #include <utils/Errors.h> | 
|  | 22 | #include <utils/Log.h> | 
| Mathias Agopian | 310f8da | 2009-05-22 01:27:01 -0700 | [diff] [blame] | 23 | #include <binder/IPCThreadState.h> | 
|  | 24 | #include <binder/IServiceManager.h> | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 25 |  | 
|  | 26 | #include <GLES/gl.h> | 
|  | 27 | #include <GLES/glext.h> | 
|  | 28 |  | 
|  | 29 | #include <hardware/hardware.h> | 
|  | 30 |  | 
|  | 31 | #include "clz.h" | 
| Mathias Agopian | db403e8 | 2012-06-18 16:47:56 -0700 | [diff] [blame] | 32 | #include "Client.h" | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 33 | #include "LayerBase.h" | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 34 | #include "SurfaceFlinger.h" | 
|  | 35 | #include "DisplayHardware/DisplayHardware.h" | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 36 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 37 | namespace android { | 
|  | 38 |  | 
|  | 39 | // --------------------------------------------------------------------------- | 
|  | 40 |  | 
| Mathias Agopian | f6679fc | 2010-08-10 18:09:09 -0700 | [diff] [blame] | 41 | int32_t LayerBase::sSequence = 1; | 
|  | 42 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 43 | LayerBase::LayerBase(SurfaceFlinger* flinger, DisplayID display) | 
|  | 44 | : dpy(display), contentDirty(false), | 
| Mathias Agopian | f6679fc | 2010-08-10 18:09:09 -0700 | [diff] [blame] | 45 | sequence(uint32_t(android_atomic_inc(&sSequence))), | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 46 | mFlinger(flinger), mFiltering(false), | 
| Mathias Agopian | a2f4e56 | 2012-04-15 23:34:59 -0700 | [diff] [blame] | 47 | mNeedsFiltering(false), | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 48 | mOrientation(0), | 
| Jamie Gennis | 8d91b42 | 2011-09-23 15:54:34 -0700 | [diff] [blame] | 49 | mPlaneOrientation(0), | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 50 | mTransactionFlags(0), | 
| Mathias Agopian | 99ce5cd | 2012-01-31 18:24:27 -0800 | [diff] [blame] | 51 | mPremultipliedAlpha(true), mName("unnamed"), mDebug(false) | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 52 | { | 
|  | 53 | const DisplayHardware& hw(flinger->graphicPlane(0).displayHardware()); | 
|  | 54 | mFlags = hw.getFlags(); | 
|  | 55 | } | 
|  | 56 |  | 
|  | 57 | LayerBase::~LayerBase() | 
|  | 58 | { | 
|  | 59 | } | 
|  | 60 |  | 
| Mathias Agopian | d129659 | 2010-03-09 19:17:47 -0800 | [diff] [blame] | 61 | void LayerBase::setName(const String8& name) { | 
|  | 62 | mName = name; | 
|  | 63 | } | 
|  | 64 |  | 
|  | 65 | String8 LayerBase::getName() const { | 
|  | 66 | return mName; | 
|  | 67 | } | 
|  | 68 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 69 | const GraphicPlane& LayerBase::graphicPlane(int dpy) const | 
|  | 70 | { | 
|  | 71 | return mFlinger->graphicPlane(dpy); | 
|  | 72 | } | 
|  | 73 |  | 
|  | 74 | GraphicPlane& LayerBase::graphicPlane(int dpy) | 
|  | 75 | { | 
|  | 76 | return mFlinger->graphicPlane(dpy); | 
|  | 77 | } | 
|  | 78 |  | 
|  | 79 | void LayerBase::initStates(uint32_t w, uint32_t h, uint32_t flags) | 
|  | 80 | { | 
|  | 81 | uint32_t layerFlags = 0; | 
|  | 82 | if (flags & ISurfaceComposer::eHidden) | 
|  | 83 | layerFlags = ISurfaceComposer::eLayerHidden; | 
|  | 84 |  | 
|  | 85 | if (flags & ISurfaceComposer::eNonPremultiplied) | 
|  | 86 | mPremultipliedAlpha = false; | 
|  | 87 |  | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 88 | mCurrentState.active.w = w; | 
|  | 89 | mCurrentState.active.h = h; | 
|  | 90 | mCurrentState.active.crop.makeInvalid(); | 
|  | 91 | mCurrentState.z = 0; | 
|  | 92 | mCurrentState.alpha = 0xFF; | 
|  | 93 | mCurrentState.flags = layerFlags; | 
|  | 94 | mCurrentState.sequence = 0; | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 95 | mCurrentState.transform.set(0, 0); | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 96 | mCurrentState.requested = mCurrentState.active; | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 97 |  | 
|  | 98 | // drawing state & current state are identical | 
|  | 99 | mDrawingState = mCurrentState; | 
|  | 100 | } | 
|  | 101 |  | 
| Mathias Agopian | ba6be54 | 2009-09-29 22:32:36 -0700 | [diff] [blame] | 102 | void LayerBase::commitTransaction() { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 103 | mDrawingState = mCurrentState; | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 104 | } | 
|  | 105 | void LayerBase::forceVisibilityTransaction() { | 
|  | 106 | // this can be called without SurfaceFlinger.mStateLock, but if we | 
|  | 107 | // can atomically increment the sequence number, it doesn't matter. | 
|  | 108 | android_atomic_inc(&mCurrentState.sequence); | 
|  | 109 | requestTransaction(); | 
|  | 110 | } | 
|  | 111 | bool LayerBase::requestTransaction() { | 
|  | 112 | int32_t old = setTransactionFlags(eTransactionNeeded); | 
|  | 113 | return ((old & eTransactionNeeded) == 0); | 
|  | 114 | } | 
|  | 115 | uint32_t LayerBase::getTransactionFlags(uint32_t flags) { | 
|  | 116 | return android_atomic_and(~flags, &mTransactionFlags) & flags; | 
|  | 117 | } | 
|  | 118 | uint32_t LayerBase::setTransactionFlags(uint32_t flags) { | 
|  | 119 | return android_atomic_or(flags, &mTransactionFlags); | 
|  | 120 | } | 
|  | 121 |  | 
| Mathias Agopian | 41b6aab | 2011-08-30 18:51:54 -0700 | [diff] [blame] | 122 | bool LayerBase::setPosition(float x, float y) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 123 | if (mCurrentState.transform.tx() == x && mCurrentState.transform.ty() == y) | 
|  | 124 | return false; | 
|  | 125 | mCurrentState.sequence++; | 
|  | 126 | mCurrentState.transform.set(x, y); | 
|  | 127 | requestTransaction(); | 
|  | 128 | return true; | 
|  | 129 | } | 
|  | 130 | bool LayerBase::setLayer(uint32_t z) { | 
|  | 131 | if (mCurrentState.z == z) | 
|  | 132 | return false; | 
|  | 133 | mCurrentState.sequence++; | 
|  | 134 | mCurrentState.z = z; | 
|  | 135 | requestTransaction(); | 
|  | 136 | return true; | 
|  | 137 | } | 
|  | 138 | bool LayerBase::setSize(uint32_t w, uint32_t h) { | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 139 | if (mCurrentState.requested.w == w && mCurrentState.requested.h == h) | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 140 | return false; | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 141 | mCurrentState.requested.w = w; | 
|  | 142 | mCurrentState.requested.h = h; | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 143 | requestTransaction(); | 
|  | 144 | return true; | 
|  | 145 | } | 
|  | 146 | bool LayerBase::setAlpha(uint8_t alpha) { | 
|  | 147 | if (mCurrentState.alpha == alpha) | 
|  | 148 | return false; | 
|  | 149 | mCurrentState.sequence++; | 
|  | 150 | mCurrentState.alpha = alpha; | 
|  | 151 | requestTransaction(); | 
|  | 152 | return true; | 
|  | 153 | } | 
|  | 154 | bool LayerBase::setMatrix(const layer_state_t::matrix22_t& matrix) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 155 | mCurrentState.sequence++; | 
|  | 156 | mCurrentState.transform.set( | 
|  | 157 | matrix.dsdx, matrix.dsdy, matrix.dtdx, matrix.dtdy); | 
|  | 158 | requestTransaction(); | 
|  | 159 | return true; | 
|  | 160 | } | 
|  | 161 | bool LayerBase::setTransparentRegionHint(const Region& transparent) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 162 | mCurrentState.sequence++; | 
|  | 163 | mCurrentState.transparentRegion = transparent; | 
|  | 164 | requestTransaction(); | 
|  | 165 | return true; | 
|  | 166 | } | 
|  | 167 | bool LayerBase::setFlags(uint8_t flags, uint8_t mask) { | 
|  | 168 | const uint32_t newFlags = (mCurrentState.flags & ~mask) | (flags & mask); | 
|  | 169 | if (mCurrentState.flags == newFlags) | 
|  | 170 | return false; | 
|  | 171 | mCurrentState.sequence++; | 
|  | 172 | mCurrentState.flags = newFlags; | 
|  | 173 | requestTransaction(); | 
|  | 174 | return true; | 
|  | 175 | } | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 176 | bool LayerBase::setCrop(const Rect& crop) { | 
| Mathias Agopian | b30c415 | 2012-05-16 18:21:32 -0700 | [diff] [blame] | 177 | if (mCurrentState.requested.crop == crop) | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 178 | return false; | 
|  | 179 | mCurrentState.sequence++; | 
| Mathias Agopian | b30c415 | 2012-05-16 18:21:32 -0700 | [diff] [blame] | 180 | mCurrentState.requested.crop = crop; | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 181 | requestTransaction(); | 
|  | 182 | return true; | 
|  | 183 | } | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 184 |  | 
|  | 185 | Rect LayerBase::visibleBounds() const | 
|  | 186 | { | 
|  | 187 | return mTransformedBounds; | 
|  | 188 | } | 
|  | 189 |  | 
|  | 190 | void LayerBase::setVisibleRegion(const Region& visibleRegion) { | 
|  | 191 | // always called from main thread | 
|  | 192 | visibleRegionScreen = visibleRegion; | 
|  | 193 | } | 
|  | 194 |  | 
|  | 195 | void LayerBase::setCoveredRegion(const Region& coveredRegion) { | 
|  | 196 | // always called from main thread | 
|  | 197 | coveredRegionScreen = coveredRegion; | 
|  | 198 | } | 
|  | 199 |  | 
|  | 200 | uint32_t LayerBase::doTransaction(uint32_t flags) | 
|  | 201 | { | 
|  | 202 | const Layer::State& front(drawingState()); | 
|  | 203 | const Layer::State& temp(currentState()); | 
|  | 204 |  | 
| Mathias Agopian | 05cec9d | 2012-05-23 14:35:49 -0700 | [diff] [blame] | 205 | // always set active to requested, unless we're asked not to | 
|  | 206 | // this is used by Layer, which special cases resizes. | 
|  | 207 | if (flags & eDontUpdateGeometryState)  { | 
|  | 208 | } else { | 
| Mathias Agopian | 7e4a587 | 2009-09-29 22:39:22 -0700 | [diff] [blame] | 209 | Layer::State& editTemp(currentState()); | 
| Mathias Agopian | b30c415 | 2012-05-16 18:21:32 -0700 | [diff] [blame] | 210 | editTemp.active = temp.requested; | 
| Mathias Agopian | 7e4a587 | 2009-09-29 22:39:22 -0700 | [diff] [blame] | 211 | } | 
| Mathias Agopian | 05cec9d | 2012-05-23 14:35:49 -0700 | [diff] [blame] | 212 |  | 
| Mathias Agopian | b30c415 | 2012-05-16 18:21:32 -0700 | [diff] [blame] | 213 | if (front.active != temp.active) { | 
| Mathias Agopian | 6656dbc | 2009-09-30 12:48:47 -0700 | [diff] [blame] | 214 | // invalidate and recompute the visible regions if needed | 
|  | 215 | flags |= Layer::eVisibleRegion; | 
| Mathias Agopian | 6656dbc | 2009-09-30 12:48:47 -0700 | [diff] [blame] | 216 | } | 
|  | 217 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 218 | if (temp.sequence != front.sequence) { | 
|  | 219 | // invalidate and recompute the visible regions if needed | 
|  | 220 | flags |= eVisibleRegion; | 
|  | 221 | this->contentDirty = true; | 
| Mathias Agopian | a2fe0a2 | 2009-09-23 18:34:53 -0700 | [diff] [blame] | 222 |  | 
| Mathias Agopian | 733189d | 2010-12-02 21:32:29 -0800 | [diff] [blame] | 223 | // we may use linear filtering, if the matrix scales us | 
|  | 224 | const uint8_t type = temp.transform.getType(); | 
|  | 225 | mNeedsFiltering = (!temp.transform.preserveRects() || | 
|  | 226 | (type >= Transform::SCALE)); | 
| Mathias Agopian | a2fe0a2 | 2009-09-23 18:34:53 -0700 | [diff] [blame] | 227 | } | 
|  | 228 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 229 | // Commit the transaction | 
| Mathias Agopian | ba6be54 | 2009-09-29 22:32:36 -0700 | [diff] [blame] | 230 | commitTransaction(); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 231 | return flags; | 
|  | 232 | } | 
|  | 233 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 234 | void LayerBase::validateVisibility(const Transform& planeTransform) | 
|  | 235 | { | 
|  | 236 | const Layer::State& s(drawingState()); | 
|  | 237 | const Transform tr(planeTransform * s.transform); | 
|  | 238 | const bool transformed = tr.transformed(); | 
| Mathias Agopian | ffcf465 | 2011-07-07 17:30:31 -0700 | [diff] [blame] | 239 | const DisplayHardware& hw(graphicPlane(0).displayHardware()); | 
|  | 240 | const uint32_t hw_h = hw.getHeight(); | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 241 | const Rect& crop(s.active.crop); | 
| Mathias Agopian | ffcf465 | 2011-07-07 17:30:31 -0700 | [diff] [blame] | 242 |  | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 243 | Rect win(s.active.w, s.active.h); | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 244 | if (!crop.isEmpty()) { | 
|  | 245 | win.intersect(crop, &win); | 
|  | 246 | } | 
| Mathias Agopian | f74e8e0 | 2012-04-16 03:14:05 -0700 | [diff] [blame] | 247 |  | 
|  | 248 | mNumVertices = 4; | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 249 | tr.transform(mVertices[0], win.left,  win.top); | 
|  | 250 | tr.transform(mVertices[1], win.left,  win.bottom); | 
|  | 251 | tr.transform(mVertices[2], win.right, win.bottom); | 
|  | 252 | tr.transform(mVertices[3], win.right, win.top); | 
| Mathias Agopian | ffcf465 | 2011-07-07 17:30:31 -0700 | [diff] [blame] | 253 | for (size_t i=0 ; i<4 ; i++) | 
|  | 254 | mVertices[i][1] = hw_h - mVertices[i][1]; | 
|  | 255 |  | 
| Glenn Kasten | 99ed224 | 2011-12-15 09:51:17 -0800 | [diff] [blame] | 256 | if (CC_UNLIKELY(transformed)) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 257 | // NOTE: here we could also punt if we have too many rectangles | 
|  | 258 | // in the transparent region | 
|  | 259 | if (tr.preserveRects()) { | 
|  | 260 | // transform the transparent region | 
|  | 261 | transparentRegionScreen = tr.transform(s.transparentRegion); | 
|  | 262 | } else { | 
|  | 263 | // transformation too complex, can't do the transparent region | 
|  | 264 | // optimization. | 
|  | 265 | transparentRegionScreen.clear(); | 
|  | 266 | } | 
|  | 267 | } else { | 
|  | 268 | transparentRegionScreen = s.transparentRegion; | 
|  | 269 | } | 
|  | 270 |  | 
|  | 271 | // cache a few things... | 
|  | 272 | mOrientation = tr.getOrientation(); | 
| Jamie Gennis | 8d91b42 | 2011-09-23 15:54:34 -0700 | [diff] [blame] | 273 | mPlaneOrientation = planeTransform.getOrientation(); | 
| Mathias Agopian | d992db3 | 2011-08-18 18:31:00 -0700 | [diff] [blame] | 274 | mTransform = tr; | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 275 | mTransformedBounds = tr.transform(win); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 276 | } | 
|  | 277 |  | 
| Mathias Agopian | 99ce5cd | 2012-01-31 18:24:27 -0800 | [diff] [blame] | 278 | void LayerBase::lockPageFlip(bool& recomputeVisibleRegions) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 279 | } | 
|  | 280 |  | 
|  | 281 | void LayerBase::unlockPageFlip( | 
| Mathias Agopian | 99ce5cd | 2012-01-31 18:24:27 -0800 | [diff] [blame] | 282 | const Transform& planeTransform, Region& outDirtyRegion) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 283 | } | 
|  | 284 |  | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 285 | void LayerBase::setGeometry(HWComposer::HWCLayerInterface& layer) | 
| Mathias Agopian | a537c0f | 2011-08-02 15:51:37 -0700 | [diff] [blame] | 286 | { | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 287 | layer.setDefaultState(); | 
| Mathias Agopian | a537c0f | 2011-08-02 15:51:37 -0700 | [diff] [blame] | 288 |  | 
|  | 289 | // this gives us only the "orientation" component of the transform | 
|  | 290 | const State& s(drawingState()); | 
|  | 291 | const uint32_t finalTransform = s.transform.getOrientation(); | 
|  | 292 | // we can only handle simple transformation | 
|  | 293 | if (finalTransform & Transform::ROT_INVALID) { | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 294 | layer.setTransform(0); | 
| Mathias Agopian | a537c0f | 2011-08-02 15:51:37 -0700 | [diff] [blame] | 295 | } else { | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 296 | layer.setTransform(finalTransform); | 
| Mathias Agopian | a537c0f | 2011-08-02 15:51:37 -0700 | [diff] [blame] | 297 | } | 
|  | 298 |  | 
|  | 299 | if (!isOpaque()) { | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 300 | layer.setBlending(mPremultipliedAlpha ? | 
|  | 301 | HWC_BLENDING_PREMULT : | 
|  | 302 | HWC_BLENDING_COVERAGE); | 
| Mathias Agopian | a537c0f | 2011-08-02 15:51:37 -0700 | [diff] [blame] | 303 | } | 
|  | 304 |  | 
|  | 305 | // scaling is already applied in mTransformedBounds | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 306 | layer.setFrame(mTransformedBounds); | 
|  | 307 | layer.setVisibleRegionScreen(visibleRegionScreen); | 
|  | 308 | layer.setCrop(mTransformedBounds.getBounds()); | 
| Mathias Agopian | a350ff9 | 2010-08-10 17:14:02 -0700 | [diff] [blame] | 309 | } | 
|  | 310 |  | 
| Mathias Agopian | 3e8b853 | 2012-05-13 20:42:01 -0700 | [diff] [blame] | 311 | void LayerBase::setPerFrameData(HWComposer::HWCLayerInterface& layer) { | 
|  | 312 | layer.setBuffer(0); | 
| Mathias Agopian | a350ff9 | 2010-08-10 17:14:02 -0700 | [diff] [blame] | 313 | } | 
|  | 314 |  | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 315 | void LayerBase::setFiltering(bool filtering) | 
|  | 316 | { | 
|  | 317 | mFiltering = filtering; | 
|  | 318 | } | 
|  | 319 |  | 
|  | 320 | bool LayerBase::getFiltering() const | 
|  | 321 | { | 
|  | 322 | return mFiltering; | 
|  | 323 | } | 
|  | 324 |  | 
| Mathias Agopian | bc7e31a | 2010-08-10 20:42:20 -0700 | [diff] [blame] | 325 | void LayerBase::draw(const Region& clip) const | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 326 | { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 327 | onDraw(clip); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 328 | } | 
|  | 329 |  | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 330 | void LayerBase::drawForSreenShot() | 
| Mathias Agopian | 74c40c0 | 2010-09-29 13:02:36 -0700 | [diff] [blame] | 331 | { | 
|  | 332 | const DisplayHardware& hw(graphicPlane(0).displayHardware()); | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 333 | setFiltering(true); | 
| Mathias Agopian | 74c40c0 | 2010-09-29 13:02:36 -0700 | [diff] [blame] | 334 | onDraw( Region(hw.bounds()) ); | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 335 | setFiltering(false); | 
| Mathias Agopian | 74c40c0 | 2010-09-29 13:02:36 -0700 | [diff] [blame] | 336 | } | 
|  | 337 |  | 
| Mathias Agopian | 010fccb | 2010-05-26 22:26:12 -0700 | [diff] [blame] | 338 | void LayerBase::clearWithOpenGL(const Region& clip, GLclampf red, | 
|  | 339 | GLclampf green, GLclampf blue, | 
|  | 340 | GLclampf alpha) const | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 341 | { | 
|  | 342 | const DisplayHardware& hw(graphicPlane(0).displayHardware()); | 
|  | 343 | const uint32_t fbHeight = hw.getHeight(); | 
| Mathias Agopian | 010fccb | 2010-05-26 22:26:12 -0700 | [diff] [blame] | 344 | glColor4f(red,green,blue,alpha); | 
| Mathias Agopian | 0a91775 | 2010-06-14 21:20:00 -0700 | [diff] [blame] | 345 |  | 
| Mathias Agopian | c492e67 | 2011-10-18 14:49:27 -0700 | [diff] [blame] | 346 | glDisable(GL_TEXTURE_EXTERNAL_OES); | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 347 | glDisable(GL_TEXTURE_2D); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 348 | glDisable(GL_BLEND); | 
| Mathias Agopian | 20f6878 | 2009-05-11 00:03:41 -0700 | [diff] [blame] | 349 |  | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 350 | glVertexPointer(2, GL_FLOAT, 0, mVertices); | 
| Mathias Agopian | f74e8e0 | 2012-04-16 03:14:05 -0700 | [diff] [blame] | 351 | glDrawArrays(GL_TRIANGLE_FAN, 0, mNumVertices); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 352 | } | 
|  | 353 |  | 
| Rebecca Schultz Zavin | 29aa74c | 2009-09-01 23:06:45 -0700 | [diff] [blame] | 354 | void LayerBase::clearWithOpenGL(const Region& clip) const | 
|  | 355 | { | 
|  | 356 | clearWithOpenGL(clip,0,0,0,0); | 
|  | 357 | } | 
|  | 358 |  | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 359 | void LayerBase::drawWithOpenGL(const Region& clip) const | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 360 | { | 
|  | 361 | const DisplayHardware& hw(graphicPlane(0).displayHardware()); | 
|  | 362 | const uint32_t fbHeight = hw.getHeight(); | 
|  | 363 | const State& s(drawingState()); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 364 |  | 
| Mathias Agopian | 4975326 | 2010-04-12 15:34:55 -0700 | [diff] [blame] | 365 | GLenum src = mPremultipliedAlpha ? GL_ONE : GL_SRC_ALPHA; | 
| Glenn Kasten | 99ed224 | 2011-12-15 09:51:17 -0800 | [diff] [blame] | 366 | if (CC_UNLIKELY(s.alpha < 0xFF)) { | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 367 | const GLfloat alpha = s.alpha * (1.0f/255.0f); | 
| Mathias Agopian | 4975326 | 2010-04-12 15:34:55 -0700 | [diff] [blame] | 368 | if (mPremultipliedAlpha) { | 
|  | 369 | glColor4f(alpha, alpha, alpha, alpha); | 
|  | 370 | } else { | 
|  | 371 | glColor4f(1, 1, 1, alpha); | 
|  | 372 | } | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 373 | glEnable(GL_BLEND); | 
|  | 374 | glBlendFunc(src, GL_ONE_MINUS_SRC_ALPHA); | 
| Mathias Agopian | 4975326 | 2010-04-12 15:34:55 -0700 | [diff] [blame] | 375 | glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 376 | } else { | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 377 | glColor4f(1, 1, 1, 1); | 
| Mathias Agopian | 4975326 | 2010-04-12 15:34:55 -0700 | [diff] [blame] | 378 | glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE); | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 379 | if (!isOpaque()) { | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 380 | glEnable(GL_BLEND); | 
|  | 381 | glBlendFunc(src, GL_ONE_MINUS_SRC_ALPHA); | 
|  | 382 | } else { | 
|  | 383 | glDisable(GL_BLEND); | 
|  | 384 | } | 
|  | 385 | } | 
|  | 386 |  | 
| Mathias Agopian | b661d66 | 2010-08-19 17:01:19 -0700 | [diff] [blame] | 387 | struct TexCoords { | 
|  | 388 | GLfloat u; | 
|  | 389 | GLfloat v; | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 390 | }; | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 391 |  | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 392 | Rect crop(s.active.w, s.active.h); | 
|  | 393 | if (!s.active.crop.isEmpty()) { | 
|  | 394 | crop = s.active.crop; | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 395 | } | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 396 | GLfloat left = GLfloat(crop.left) / GLfloat(s.active.w); | 
|  | 397 | GLfloat top = GLfloat(crop.top) / GLfloat(s.active.h); | 
|  | 398 | GLfloat right = GLfloat(crop.right) / GLfloat(s.active.w); | 
|  | 399 | GLfloat bottom = GLfloat(crop.bottom) / GLfloat(s.active.h); | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 400 |  | 
| Mathias Agopian | b661d66 | 2010-08-19 17:01:19 -0700 | [diff] [blame] | 401 | TexCoords texCoords[4]; | 
| Jamie Gennis | f15a83f | 2012-05-10 20:43:55 -0700 | [diff] [blame] | 402 | texCoords[0].u = left; | 
|  | 403 | texCoords[0].v = top; | 
|  | 404 | texCoords[1].u = left; | 
|  | 405 | texCoords[1].v = bottom; | 
|  | 406 | texCoords[2].u = right; | 
|  | 407 | texCoords[2].v = bottom; | 
|  | 408 | texCoords[3].u = right; | 
|  | 409 | texCoords[3].v = top; | 
|  | 410 | for (int i = 0; i < 4; i++) { | 
|  | 411 | texCoords[i].v = 1.0f - texCoords[i].v; | 
|  | 412 | } | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 413 |  | 
|  | 414 | glEnableClientState(GL_TEXTURE_COORD_ARRAY); | 
|  | 415 | glVertexPointer(2, GL_FLOAT, 0, mVertices); | 
|  | 416 | glTexCoordPointer(2, GL_FLOAT, 0, texCoords); | 
| Mathias Agopian | f74e8e0 | 2012-04-16 03:14:05 -0700 | [diff] [blame] | 417 | glDrawArrays(GL_TRIANGLE_FAN, 0, mNumVertices); | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 418 |  | 
| Mathias Agopian | 78fd501 | 2010-04-20 14:51:04 -0700 | [diff] [blame] | 419 | glDisableClientState(GL_TEXTURE_COORD_ARRAY); | 
| Mathias Agopian | c492e67 | 2011-10-18 14:49:27 -0700 | [diff] [blame] | 420 | glDisable(GL_BLEND); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 421 | } | 
|  | 422 |  | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 423 | void LayerBase::dump(String8& result, char* buffer, size_t SIZE) const | 
|  | 424 | { | 
|  | 425 | const Layer::State& s(drawingState()); | 
| Mathias Agopian | c95dbdc | 2012-02-05 00:19:27 -0800 | [diff] [blame] | 426 |  | 
|  | 427 | snprintf(buffer, SIZE, | 
|  | 428 | "+ %s %p (%s)\n", | 
|  | 429 | getTypeId(), this, getName().string()); | 
|  | 430 | result.append(buffer); | 
|  | 431 |  | 
| Mathias Agopian | 82d7ab6 | 2012-01-19 18:34:40 -0800 | [diff] [blame] | 432 | s.transparentRegion.dump(result, "transparentRegion"); | 
|  | 433 | transparentRegionScreen.dump(result, "transparentRegionScreen"); | 
|  | 434 | visibleRegionScreen.dump(result, "visibleRegionScreen"); | 
| Mathias Agopian | c95dbdc | 2012-02-05 00:19:27 -0800 | [diff] [blame] | 435 |  | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 436 | snprintf(buffer, SIZE, | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 437 | "      " | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 438 | "z=%9d, pos=(%g,%g), size=(%4d,%4d), crop=(%4d,%4d,%4d,%4d), " | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 439 | "isOpaque=%1d, needsDithering=%1d, invalidate=%1d, " | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 440 | "alpha=0x%02x, flags=0x%08x, tr=[%.2f, %.2f][%.2f, %.2f]\n", | 
| Mathias Agopian | 93ffb86 | 2012-05-16 17:07:49 -0700 | [diff] [blame] | 441 | s.z, s.transform.tx(), s.transform.ty(), s.active.w, s.active.h, | 
|  | 442 | s.active.crop.left, s.active.crop.top, | 
|  | 443 | s.active.crop.right, s.active.crop.bottom, | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 444 | isOpaque(), needsDithering(), contentDirty, | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 445 | s.alpha, s.flags, | 
|  | 446 | s.transform[0][0], s.transform[0][1], | 
|  | 447 | s.transform[1][0], s.transform[1][1]); | 
|  | 448 | result.append(buffer); | 
|  | 449 | } | 
| Mathias Agopian | 54ba51d | 2009-10-26 20:12:37 -0700 | [diff] [blame] | 450 |  | 
| Mathias Agopian | 25e66fc | 2012-01-28 22:31:55 -0800 | [diff] [blame] | 451 | void LayerBase::shortDump(String8& result, char* scratch, size_t size) const { | 
| Mathias Agopian | 48b888a | 2011-01-19 16:15:53 -0800 | [diff] [blame] | 452 | LayerBase::dump(result, scratch, size); | 
|  | 453 | } | 
|  | 454 |  | 
| Mathias Agopian | 25e66fc | 2012-01-28 22:31:55 -0800 | [diff] [blame] | 455 | void LayerBase::dumpStats(String8& result, char* scratch, size_t SIZE) const { | 
|  | 456 | } | 
|  | 457 |  | 
|  | 458 | void LayerBase::clearStats() { | 
| Mathias Agopian | 82d7ab6 | 2012-01-19 18:34:40 -0800 | [diff] [blame] | 459 | } | 
| Mathias Agopian | 48b888a | 2011-01-19 16:15:53 -0800 | [diff] [blame] | 460 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 461 | // --------------------------------------------------------------------------- | 
|  | 462 |  | 
| Mathias Agopian | 631f358 | 2010-05-25 17:51:34 -0700 | [diff] [blame] | 463 | int32_t LayerBaseClient::sIdentity = 1; | 
| Mathias Agopian | 2e12324 | 2009-06-23 20:06:46 -0700 | [diff] [blame] | 464 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 465 | LayerBaseClient::LayerBaseClient(SurfaceFlinger* flinger, DisplayID display, | 
| Mathias Agopian | 96f0819 | 2010-06-02 23:28:45 -0700 | [diff] [blame] | 466 | const sp<Client>& client) | 
| Mathias Agopian | a1f47b9 | 2011-02-15 19:01:06 -0800 | [diff] [blame] | 467 | : LayerBase(flinger, display), | 
|  | 468 | mHasSurface(false), | 
|  | 469 | mClientRef(client), | 
| Mathias Agopian | 948d69f | 2010-03-08 19:29:09 -0800 | [diff] [blame] | 470 | mIdentity(uint32_t(android_atomic_inc(&sIdentity))) | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 471 | { | 
| Mathias Agopian | 076b1cc | 2009-04-10 14:24:30 -0700 | [diff] [blame] | 472 | } | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 473 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 474 | LayerBaseClient::~LayerBaseClient() | 
|  | 475 | { | 
| Mathias Agopian | b7e930d | 2010-06-01 15:12:58 -0700 | [diff] [blame] | 476 | sp<Client> c(mClientRef.promote()); | 
| Mathias Agopian | 96f0819 | 2010-06-02 23:28:45 -0700 | [diff] [blame] | 477 | if (c != 0) { | 
| Mathias Agopian | b7e930d | 2010-06-01 15:12:58 -0700 | [diff] [blame] | 478 | c->detachLayer(this); | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 479 | } | 
|  | 480 | } | 
|  | 481 |  | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 482 | sp<ISurface> LayerBaseClient::createSurface() | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 483 | { | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 484 | class BSurface : public BnSurface, public LayerCleaner { | 
|  | 485 | virtual sp<ISurfaceTexture> getSurfaceTexture() const { return 0; } | 
|  | 486 | public: | 
|  | 487 | BSurface(const sp<SurfaceFlinger>& flinger, | 
|  | 488 | const sp<LayerBaseClient>& layer) | 
|  | 489 | : LayerCleaner(flinger, layer) { } | 
|  | 490 | }; | 
|  | 491 | sp<ISurface> sur(new BSurface(mFlinger, this)); | 
|  | 492 | return sur; | 
|  | 493 | } | 
|  | 494 |  | 
|  | 495 | sp<ISurface> LayerBaseClient::getSurface() | 
|  | 496 | { | 
|  | 497 | sp<ISurface> s; | 
| Mathias Agopian | 076b1cc | 2009-04-10 14:24:30 -0700 | [diff] [blame] | 498 | Mutex::Autolock _l(mLock); | 
| Mathias Agopian | a1f47b9 | 2011-02-15 19:01:06 -0800 | [diff] [blame] | 499 |  | 
|  | 500 | LOG_ALWAYS_FATAL_IF(mHasSurface, | 
|  | 501 | "LayerBaseClient::getSurface() has already been called"); | 
|  | 502 |  | 
|  | 503 | mHasSurface = true; | 
|  | 504 | s = createSurface(); | 
|  | 505 | mClientSurfaceBinder = s->asBinder(); | 
| Mathias Agopian | 076b1cc | 2009-04-10 14:24:30 -0700 | [diff] [blame] | 506 | return s; | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 507 | } | 
|  | 508 |  | 
| Mathias Agopian | 0d15612 | 2011-01-25 20:17:45 -0800 | [diff] [blame] | 509 | wp<IBinder> LayerBaseClient::getSurfaceBinder() const { | 
|  | 510 | return mClientSurfaceBinder; | 
|  | 511 | } | 
|  | 512 |  | 
| Jamie Gennis | 582270d | 2011-08-17 18:19:00 -0700 | [diff] [blame] | 513 | wp<IBinder> LayerBaseClient::getSurfaceTextureBinder() const { | 
|  | 514 | return 0; | 
|  | 515 | } | 
|  | 516 |  | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 517 | void LayerBaseClient::dump(String8& result, char* buffer, size_t SIZE) const | 
|  | 518 | { | 
|  | 519 | LayerBase::dump(result, buffer, SIZE); | 
|  | 520 |  | 
| Mathias Agopian | b7e930d | 2010-06-01 15:12:58 -0700 | [diff] [blame] | 521 | sp<Client> client(mClientRef.promote()); | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 522 | snprintf(buffer, SIZE, | 
| Mathias Agopian | 96f0819 | 2010-06-02 23:28:45 -0700 | [diff] [blame] | 523 | "      client=%p, identity=%u\n", | 
| Mathias Agopian | 96f0819 | 2010-06-02 23:28:45 -0700 | [diff] [blame] | 524 | client.get(), getIdentity()); | 
| Mathias Agopian | 1b5e102 | 2010-04-20 17:55:49 -0700 | [diff] [blame] | 525 |  | 
|  | 526 | result.append(buffer); | 
|  | 527 | } | 
|  | 528 |  | 
| Mathias Agopian | 48b888a | 2011-01-19 16:15:53 -0800 | [diff] [blame] | 529 |  | 
|  | 530 | void LayerBaseClient::shortDump(String8& result, char* scratch, size_t size) const | 
|  | 531 | { | 
|  | 532 | LayerBaseClient::dump(result, scratch, size); | 
|  | 533 | } | 
|  | 534 |  | 
| Mathias Agopian | 076b1cc | 2009-04-10 14:24:30 -0700 | [diff] [blame] | 535 | // --------------------------------------------------------------------------- | 
|  | 536 |  | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 537 | LayerBaseClient::LayerCleaner::LayerCleaner(const sp<SurfaceFlinger>& flinger, | 
|  | 538 | const sp<LayerBaseClient>& layer) | 
|  | 539 | : mFlinger(flinger), mLayer(layer) { | 
| Mathias Agopian | 076b1cc | 2009-04-10 14:24:30 -0700 | [diff] [blame] | 540 | } | 
|  | 541 |  | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 542 | LayerBaseClient::LayerCleaner::~LayerCleaner() { | 
| Mathias Agopian | 9a11206 | 2009-04-17 19:36:26 -0700 | [diff] [blame] | 543 | // destroy client resources | 
| Mathias Agopian | a67932f | 2011-04-20 14:20:59 -0700 | [diff] [blame] | 544 | mFlinger->destroySurface(mLayer); | 
| Mathias Agopian | b5b7f26 | 2010-05-07 15:58:44 -0700 | [diff] [blame] | 545 | } | 
|  | 546 |  | 
| The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 547 | // --------------------------------------------------------------------------- | 
|  | 548 |  | 
|  | 549 | }; // namespace android |