BootAnimation.cpp revision f67b317c102db72182046ec2a2cffe2331017709
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
17ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood#define LOG_NDEBUG 0
18bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian#define LOG_TAG "BootAnimation"
19bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdint.h>
219748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi#include <sys/inotify.h>
229748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi#include <sys/poll.h>
239748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi#include <sys/stat.h>
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/types.h>
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <math.h>
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <fcntl.h>
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/misc.h>
28b4d5a72691846bce5779cc3db056f09a5031d7ccMathias Agopian#include <signal.h>
29bb94f3107ea567ddf67bed4617c3bcbd602538dfElliott Hughes#include <time.h>
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
31bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks#include <cutils/properties.h>
32bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
33b13b9bdad2baf6ad1ec2e56b6b7598fa20f55fc4Mathias Agopian#include <androidfw/AssetManager.h>
34ac31a3b8b09aba1c5ebc73f0cf65cac2210aa6b7Mathias Agopian#include <binder/IPCThreadState.h>
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Atomic.h>
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Errors.h>
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Log.h>
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/PixelFormat.h>
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/Rect.h>
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/Region.h>
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/DisplayInfo.h>
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
440b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown#include <gui/ISurfaceComposer.h>
458335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian#include <gui/Surface.h>
468335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian#include <gui/SurfaceComposerClient.h>
47000479f9e325b4e426a67033abd92d47da412725Mathias Agopian
48cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe// TODO: Fix Skia.
49cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe#pragma GCC diagnostic push
50cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe#pragma GCC diagnostic ignored "-Wunused-parameter"
51eece0dda56ae29fff6e9003df97594f6ac50b6e2Derek Sollenberger#include <SkBitmap.h>
528898c16987831db47542a92151ed7209e22268aaMatt Sarett#include <SkImage.h>
53eece0dda56ae29fff6e9003df97594f6ac50b6e2Derek Sollenberger#include <SkStream.h>
54cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe#pragma GCC diagnostic pop
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/gl.h>
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/glext.h>
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <EGL/eglext.h>
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "BootAnimation.h"
61d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch#include "audioplay.h"
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
659748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char OEM_BOOTANIMATION_FILE[] = "/oem/media/bootanimation.zip";
669748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_BOOTANIMATION_FILE[] = "/system/media/bootanimation.zip";
679748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_ENCRYPTED_BOOTANIMATION_FILE[] = "/system/media/bootanimation-encrypted.zip";
689748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_DATA_DIR_PATH[] = "/data/system";
699748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_TIME_DIR_NAME[] = "time";
709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_TIME_DIR_PATH[] = "/data/system/time";
710e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const char CLOCK_FONT_ASSET[] = "images/clock_font.png";
720e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const char CLOCK_FONT_ZIP_NAME[] = "clock_font.png";
739748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char LAST_TIME_CHANGED_FILE_NAME[] = "last_time_change";
749748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char LAST_TIME_CHANGED_FILE_PATH[] = "/data/system/time/last_time_change";
759748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char ACCURATE_TIME_FLAG_FILE_NAME[] = "time_is_accurate";
769748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char ACCURATE_TIME_FLAG_FILE_PATH[] = "/data/system/time/time_is_accurate";
779676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi// Java timestamp format. Don't show the clock if the date is before 2000-01-01 00:00:00.
789676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchistatic const long long ACCURATE_TIME_EPOCH = 946684800000;
790e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr char FONT_BEGIN_CHAR = ' ';
800e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr char FONT_END_CHAR = '~' + 1;
810e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_CHARS = FONT_END_CHAR - FONT_BEGIN_CHAR + 1;
820e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_COLS = 16;
830e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_ROWS = FONT_NUM_CHARS / FONT_NUM_COLS;
840e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const int TEXT_CENTER_VALUE = INT_MAX;
850e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const int TEXT_MISSING_VALUE = INT_MIN;
869748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char EXIT_PROP_NAME[] = "service.bootanim.exit";
87305087991d1b88d98c5e5e03fcf5a6a6e93ad356Geoffrey Pitschstatic const char PLAY_SOUND_PROP_NAME[] = "persist.sys.bootanim.play_sound";
88afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamathstatic const int ANIM_ENTRY_NAME_MAX = 256;
890e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t TEXT_POS_LEN_MAX = 16;
90290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitschstatic const char BOOT_COMPLETED_PROP_NAME[] = "sys.boot_completed";
91290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitschstatic const char BOOTREASON_PROP_NAME[] = "ro.boot.bootreason";
92290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch// bootreasons list in "system/core/bootstat/bootstat.cpp".
93290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitschstatic const std::vector<std::string> PLAY_SOUND_BOOTREASON_BLACKLIST {
94290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch  "kernel_panic",
95290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch  "Panic",
96290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch  "Watchdog",
97290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch};
98afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1019748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::BootAnimation() : Thread(false), mClockEnabled(true), mTimeIsAccurate(false),
1029748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread(NULL) {
103627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    mSession = new SurfaceComposerClient();
104290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch
105290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    // If the system has already booted, the animation is not being used for a boot.
106290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    mSystemBoot = !property_get_bool(BOOT_COMPLETED_PROP_NAME, 0);
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1099748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::~BootAnimation() {}
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid BootAnimation::onFirstRef() {
112bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    status_t err = mSession->linkToComposerDeath(this);
1133762c311729fe9f3af085c14c5c1fb471d994c03Steve Block    ALOGE_IF(err, "linkToComposerDeath failed (%s) ", strerror(-err));
1148434c5369304e639efe8eab368ca410c589d87c2Mathias Agopian    if (err == NO_ERROR) {
115bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian        run("BootAnimation", PRIORITY_DISPLAY);
116bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    }
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
119bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopiansp<SurfaceComposerClient> BootAnimation::session() const {
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mSession;
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
123bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
124afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamathvoid BootAnimation::binderDied(const wp<IBinder>&)
125bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian{
126bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // woah, surfaceflinger died!
1275baa3a62a97544669fba6d65a11c07f252e654ddSteve Block    ALOGD("SurfaceFlinger died, exiting...");
128bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
129bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // calling requestExit() is not enough here because the Surface code
130bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // might be blocked on a condition variable that will never be updated.
131bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    kill( getpid(), SIGKILL );
132bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    requestExit();
133d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch    audioplay::destroy();
134bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian}
135bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t BootAnimation::initTexture(Texture* texture, AssetManager& assets,
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const char* name) {
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Asset* asset = assets.open(name, Asset::ACCESS_BUFFER);
139cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe    if (asset == NULL)
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_INIT;
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    SkBitmap bitmap;
1428898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkData> data = SkData::MakeWithoutCopy(asset->getBuffer(false),
1438898c16987831db47542a92151ed7209e22268aaMatt Sarett            asset->getLength());
1448898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkImage> image = SkImage::MakeFromEncoded(data);
1458898c16987831db47542a92151ed7209e22268aaMatt Sarett    image->asLegacyBitmap(&bitmap, SkImage::kRO_LegacyBitmapMode);
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    asset->close();
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete asset;
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // ensure we can call getPixels(). No need to call unlock, since the
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // bitmap will go out of scope when we return from this method.
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bitmap.lockPixels();
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int w = bitmap.width();
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int h = bitmap.height();
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const void* p = bitmap.getPixels();
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    GLint crop[4] = { 0, h, w, -h };
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    texture->w = w;
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    texture->h = h;
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &texture->name);
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, texture->name);
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
16442a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed    switch (bitmap.colorType()) {
16542a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kAlpha_8_SkColorType:
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, w, h, 0, GL_ALPHA,
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_BYTE, p);
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
16942a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kARGB_4444_SkColorType:
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_SHORT_4_4_4_4, p);
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
17342a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kN32_SkColorType:
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_BYTE, p);
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
17742a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kRGB_565_SkColorType:
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB,
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_SHORT_5_6_5, p);
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        default:
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, crop);
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
1900e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1940e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatus_t BootAnimation::initTexture(FileMap* map, int* width, int* height)
195a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
196a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    SkBitmap bitmap;
197f67b317c102db72182046ec2a2cffe2331017709Damien Bargiacchi    sk_sp<SkData> data = SkData::MakeWithoutCopy(map->getDataPtr(),
198f67b317c102db72182046ec2a2cffe2331017709Damien Bargiacchi            map->getDataLength());
1998898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkImage> image = SkImage::MakeFromEncoded(data);
2008898c16987831db47542a92151ed7209e22268aaMatt Sarett    image->asLegacyBitmap(&bitmap, SkImage::kRO_LegacyBitmapMode);
201a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2020c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // FileMap memory is never released until application exit.
2030c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // Release it now as the texture is already loaded and the memory used for
2040c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // the packed resource can be released.
2050e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    delete map;
2060c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko
207a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // ensure we can call getPixels(). No need to call unlock, since the
208a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // bitmap will go out of scope when we return from this method.
209a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    bitmap.lockPixels();
210a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
211a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const int w = bitmap.width();
212a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const int h = bitmap.height();
213a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const void* p = bitmap.getPixels();
214a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
215a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    GLint crop[4] = { 0, h, w, -h };
216a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    int tw = 1 << (31 - __builtin_clz(w));
217a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    int th = 1 << (31 - __builtin_clz(h));
218a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    if (tw < w) tw <<= 1;
219a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    if (th < h) th <<= 1;
220a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
22142a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed    switch (bitmap.colorType()) {
22242a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kN32_SkColorType:
223271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            if (!mUseNpotTextures && (tw != w || th != h)) {
224a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, tw, th, 0, GL_RGBA,
225a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_BYTE, 0);
226a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexSubImage2D(GL_TEXTURE_2D, 0,
227a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        0, 0, w, h, GL_RGBA, GL_UNSIGNED_BYTE, p);
228a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            } else {
229271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
230a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_BYTE, p);
231a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
232a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
233a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
23442a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kRGB_565_SkColorType:
235271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            if (!mUseNpotTextures && (tw != w || th != h)) {
236a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, tw, th, 0, GL_RGB,
237a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_SHORT_5_6_5, 0);
238a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexSubImage2D(GL_TEXTURE_2D, 0,
239a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        0, 0, w, h, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, p);
240a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            } else {
241271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB,
242a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_SHORT_5_6_5, p);
243a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
244a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
245a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        default:
246a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
247a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
248a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
249a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, crop);
250a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *width = w;
2520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *height = h;
2530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
254a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    return NO_ERROR;
255a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian}
256a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t BootAnimation::readyToRun() {
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mAssets.addDefaultAssets();
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2600b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    sp<IBinder> dtoken(SurfaceComposerClient::getBuiltInDisplay(
2610b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown            ISurfaceComposer::eDisplayIdMain));
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    DisplayInfo dinfo;
2630b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    status_t status = SurfaceComposerClient::getDisplayInfo(dtoken, &dinfo);
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (status)
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return -1;
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // create the native surface
2680b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    sp<SurfaceControl> control = session()->createSurface(String8("BootAnimation"),
2690b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown            dinfo.w, dinfo.h, PIXEL_FORMAT_RGB_565);
270439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian
271439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    SurfaceComposerClient::openGlobalTransaction();
27217f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    control->setLayer(0x40000000);
273439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    SurfaceComposerClient::closeGlobalTransaction();
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
27517f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    sp<Surface> s = control->getSurface();
27617f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize opengl and egl
278738b9a45399485893fa9541fae19c47e01bc4af4Mathias Agopian    const EGLint attribs[] = {
2791b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_RED_SIZE,   8,
2801b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_GREEN_SIZE, 8,
2811b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_BLUE_SIZE,  8,
282a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            EGL_DEPTH_SIZE, 0,
283a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            EGL_NONE
284738b9a45399485893fa9541fae19c47e01bc4af4Mathias Agopian    };
285cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe    EGLint w, h;
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLint numConfigs;
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLConfig config;
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLSurface surface;
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLContext context;
290627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
292627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
293627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    eglInitialize(display, 0, 0);
2941b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian    eglChooseConfig(display, attribs, &config, 1, &numConfigs);
2951473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    surface = eglCreateWindowSurface(display, config, s.get(), NULL);
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    context = eglCreateContext(display, config, NULL, NULL);
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglQuerySurface(display, surface, EGL_WIDTH, &w);
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglQuerySurface(display, surface, EGL_HEIGHT, &h);
299a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
300abac01052c1c40f8582fe9f010c2efe6013e25c6Mathias Agopian    if (eglMakeCurrent(display, surface, surface, context) == EGL_FALSE)
301abac01052c1c40f8582fe9f010c2efe6013e25c6Mathias Agopian        return NO_INIT;
302a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDisplay = display;
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mContext = context;
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSurface = surface;
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mWidth = w;
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mHeight = h;
30817f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    mFlingerSurfaceControl = control;
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFlingerSurface = s;
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
311c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes    // If the device has encryption turned on or is in process
312bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    // of being encrypted we show the encrypted boot animation.
313bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    char decrypt[PROPERTY_VALUE_MAX];
314bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    property_get("vold.decrypt", decrypt, "");
315bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
316bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    bool encryptedAnimation = atoi(decrypt) != 0 || !strcmp("trigger_restart_min_framework", decrypt);
317bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
31839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (encryptedAnimation && (access(SYSTEM_ENCRYPTED_BOOTANIMATION_FILE, R_OK) == 0)) {
31939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        mZipFileName = SYSTEM_ENCRYPTED_BOOTANIMATION_FILE;
32039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
32139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    else if (access(OEM_BOOTANIMATION_FILE, R_OK) == 0) {
32239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        mZipFileName = OEM_BOOTANIMATION_FILE;
32339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
32439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    else if (access(SYSTEM_BOOTANIMATION_FILE, R_OK) == 0) {
32539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        mZipFileName = SYSTEM_BOOTANIMATION_FILE;
326bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    }
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
330a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopianbool BootAnimation::threadLoop()
331a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
332a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    bool r;
333afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    // We have no bootanimation file, so we use the stock android logo
334afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    // animation.
33539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (mZipFileName.isEmpty()) {
336a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        r = android();
337a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    } else {
338a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        r = movie();
339a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
340a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglMakeCurrent(mDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglDestroyContext(mDisplay, mContext);
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglDestroySurface(mDisplay, mSurface);
3446cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    mFlingerSurface.clear();
34517f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    mFlingerSurfaceControl.clear();
346627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    eglTerminate(mDisplay);
3473eee9fbb6df1dbbcefe1f0dfdf1295c95ddf3279Sai Kiran Korwar    eglReleaseThread();
348627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    IPCThreadState::self()->stopProcess();
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return r;
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
352a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopianbool BootAnimation::android()
353a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
354b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    initTexture(&mAndroid[0], mAssets, "images/android-logo-mask.png");
355b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    initTexture(&mAndroid[1], mAssets, "images/android-logo-shine.png");
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear screen
358a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glShadeModel(GL_FLAT);
359b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glDisable(GL_DITHER);
360b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glDisable(GL_SCISSOR_TEST);
36159f19e48c1c043bb9debdc35d166e397e2125d33Mathias Agopian    glClearColor(0,0,0,1);
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glClear(GL_COLOR_BUFFER_BIT);
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglSwapBuffers(mDisplay, mSurface);
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
365a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glEnable(GL_TEXTURE_2D);
366a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
367a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
368b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const GLint xc = (mWidth  - mAndroid[0].w) / 2;
369b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const GLint yc = (mHeight - mAndroid[0].h) / 2;
370b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const Rect updateRect(xc, yc, xc + mAndroid[0].w, yc + mAndroid[0].h);
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glScissor(updateRect.left, mHeight - updateRect.bottom, updateRect.width(),
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            updateRect.height());
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
375b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    // Blend state
376b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
377b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
378b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t startTime = systemTime();
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    do {
3811379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        nsecs_t now = systemTime();
3821379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        double time = now - startTime;
383b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        float t = 4.0f * float(time / us2ns(16667)) / mAndroid[1].w;
384b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        GLint offset = (1 - (t - floorf(t))) * mAndroid[1].w;
385b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        GLint x = xc - offset;
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3878166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glDisable(GL_SCISSOR_TEST);
3888166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
3898166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian
3908166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glEnable(GL_SCISSOR_TEST);
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisable(GL_BLEND);
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mAndroid[1].name);
393b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(x,                 yc, 0, mAndroid[1].w, mAndroid[1].h);
394b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(x + mAndroid[1].w, yc, 0, mAndroid[1].w, mAndroid[1].h);
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
396b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glEnable(GL_BLEND);
397b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glBindTexture(GL_TEXTURE_2D, mAndroid[0].name);
398b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(xc, yc, 0, mAndroid[0].w, mAndroid[0].h);
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
400627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian        EGLBoolean res = eglSwapBuffers(mDisplay, mSurface);
401627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian        if (res == EGL_FALSE)
402627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            break;
403627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
4041379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        // 12fps: don't animate too fast to preserve CPU
4051379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        const nsecs_t sleepTime = 83333 - ns2us(systemTime() - now);
4061379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        if (sleepTime > 0)
407a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            usleep(sleepTime);
408d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
409d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        checkExit();
4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } while (!exitPending());
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mAndroid[0].name);
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mAndroid[1].name);
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return false;
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
417d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hestervoid BootAnimation::checkExit() {
418d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    // Allow surface flinger to gracefully request shutdown
419d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    char value[PROPERTY_VALUE_MAX];
420d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    property_get(EXIT_PROP_NAME, value, "0");
421d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    int exitnow = atoi(value);
422d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    if (exitnow) {
423d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        requestExit();
424d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    }
425d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester}
426d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
4270e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchibool BootAnimation::validClock(const Animation::Part& part) {
4280e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return part.clockPosX != TEXT_MISSING_VALUE && part.clockPosY != TEXT_MISSING_VALUE;
4290e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4300e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4310e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchibool parseTextCoord(const char* str, int* dest) {
4320e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (strcmp("c", str) == 0) {
4330e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *dest = TEXT_CENTER_VALUE;
4340e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return true;
4350e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    char* end;
4380e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int val = (int) strtol(str, &end, 0);
4390e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (end == str || *end != '\0' || val == INT_MAX || val == INT_MIN) {
4400e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return false;
4410e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *dest = val;
4430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return true;
4440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// Parse two position coordinates. If only string is non-empty, treat it as the y value.
4470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid parsePosition(const char* str1, const char* str2, int* x, int* y) {
4480e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    bool success = false;
4490e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (strlen(str1) == 0) {  // No values were specified
4500e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        // success = false
4510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (strlen(str2) == 0) {  // we have only one value
4520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (parseTextCoord(str1, y)) {
4530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            *x = TEXT_CENTER_VALUE;
4540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            success = true;
4550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        }
4560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else {
4570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (parseTextCoord(str1, x) && parseTextCoord(str2, y)) {
4580e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            success = true;
4590e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        }
4600e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (!success) {
4630e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = TEXT_MISSING_VALUE;
4640e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = TEXT_MISSING_VALUE;
4650e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4660e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4670e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
468083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// Parse a color represented as an HTML-style 'RRGGBB' string: each pair of
469083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// characters in str is a hex number in [0, 255], which are converted to
470083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// floating point values in the range [0.0, 1.0] and placed in the
471083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// corresponding elements of color.
472083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall//
473083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// If the input string isn't valid, parseColor returns false and color is
474083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// left unchanged.
475083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hallstatic bool parseColor(const char str[7], float color[3]) {
476083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    float tmpColor[3];
477083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    for (int i = 0; i < 3; i++) {
478083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        int val = 0;
479083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        for (int j = 0; j < 2; j++) {
480083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            val *= 16;
481083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            char c = str[2*i + j];
482083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            if      (c >= '0' && c <= '9') val += c - '0';
483083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else if (c >= 'A' && c <= 'F') val += (c - 'A') + 10;
484083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else if (c >= 'a' && c <= 'f') val += (c - 'a') + 10;
485083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else                           return false;
486083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        }
487083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        tmpColor[i] = static_cast<float>(val) / 255.0f;
488083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    }
489083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    memcpy(color, tmpColor, sizeof(tmpColor));
490083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    return true;
491083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall}
492083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
49339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
49439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyystatic bool readFile(ZipFileRO* zip, const char* name, String8& outString)
495a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
49639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    ZipEntryRO entry = zip->findEntryByName(name);
497ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    ALOGE_IF(!entry, "couldn't find %s", name);
498ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    if (!entry) {
499afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        return false;
500afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    }
501a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
50239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    FileMap* entryMap = zip->createEntryFileMap(entry);
50339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    zip->releaseEntry(entry);
504ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    ALOGE_IF(!entryMap, "entryMap is null");
505ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    if (!entryMap) {
506a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        return false;
507a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
508a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
509ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    outString.setTo((char const*)entryMap->getDataPtr(), entryMap->getDataLength());
510688ff4cf0f93d1a886437884f23874e5595d59a3Narayan Kamath    delete entryMap;
511ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    return true;
512ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood}
513ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
5140e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// The font image should be a 96x2 array of character images.  The
5150e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// columns are the printable ASCII characters 0x20 - 0x7f.  The
5160e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// top row is regular text; the bottom row is bold.
5170e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatus_t BootAnimation::initFont(Font* font, const char* fallback) {
5180e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    status_t status = NO_ERROR;
519a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5200e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (font->map != nullptr) {
5210e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glGenTextures(1, &font->texture.name);
5220e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glBindTexture(GL_TEXTURE_2D, font->texture.name);
523a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5240e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        status = initTexture(font->map, &font->texture.w, &font->texture.h);
5250e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5260e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
5270e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
5280e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
5290e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
5300e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (fallback != nullptr) {
5310e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        status = initTexture(&font->texture, mAssets, fallback);
5320e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else {
5330e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return NO_INIT;
534a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
535a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (status == NO_ERROR) {
5370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        font->char_width = font->texture.w / FONT_NUM_COLS;
5380e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        font->char_height = font->texture.h / FONT_NUM_ROWS / 2;  // There are bold and regular rows
5390e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
540a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5410e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return status;
5420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
543a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid BootAnimation::drawText(const char* str, const Font& font, bool bold, int* x, int* y) {
5450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    glEnable(GL_BLEND);  // Allow us to draw on top of the animation
5460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    glBindTexture(GL_TEXTURE_2D, font.texture.name);
5470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5480e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    const int len = strlen(str);
5490e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    const int strWidth = font.char_width * len;
5500e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (*x == TEXT_CENTER_VALUE) {
5520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = (mWidth - strWidth) / 2;
5530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (*x < 0) {
5540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = mWidth + *x - strWidth;
5550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
5560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (*y == TEXT_CENTER_VALUE) {
5570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = (mHeight - font.char_height) / 2;
5580e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (*y < 0) {
5590e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = mHeight + *y - font.char_height;
560a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
561a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int cropRect[4] = { 0, 0, font.char_width, -font.char_height };
563a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5640e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    for (int i = 0; i < len; i++) {
5650e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char c = str[i];
566a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5670e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (c < FONT_BEGIN_CHAR || c > FONT_END_CHAR) {
5680e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            c = '?';
569a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        }
570a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
571a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        // Crop the texture to only the pixels in the current glyph
5720e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int charPos = (c - FONT_BEGIN_CHAR);  // Position in the list of valid characters
5730e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int row = charPos / FONT_NUM_COLS;
5740e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int col = charPos % FONT_NUM_COLS;
5750e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[0] = col * font.char_width;  // Left of column
5760e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[1] = row * font.char_height * 2; // Top of row
5770e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        // Move down to bottom of regular (one char_heigh) or bold (two char_heigh) line
5780e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[1] += bold ? 2 * font.char_height : font.char_height;
579a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, cropRect);
580a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5810e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glDrawTexiOES(*x, *y, 0, font.char_width, font.char_height);
582a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5830e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x += font.char_width;
584a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
585a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
586a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glDisable(GL_BLEND);  // Return to the animation's default behaviour
587a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glBindTexture(GL_TEXTURE_2D, 0);
588a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi}
589a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5900e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// We render 24 hour time.
5910e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid BootAnimation::drawClock(const Font& font, const int xPos, const int yPos) {
5920e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    static constexpr char TIME_FORMAT[] = "%H:%M";
5930e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    static constexpr int TIME_LENGTH = 6;
5940e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5950e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    time_t rawtime;
5960e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    time(&rawtime);
5970e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    struct tm* timeInfo = localtime(&rawtime);
5980e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5990e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    char timeBuff[TIME_LENGTH];
6000e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    size_t length = strftime(timeBuff, TIME_LENGTH, TIME_FORMAT, timeInfo);
6010e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6020e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (length != TIME_LENGTH - 1) {
6030e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        ALOGE("Couldn't format time; abandoning boot animation clock");
6040e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        mClockEnabled = false;
6050e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return;
6060e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
6070e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6080e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int x = xPos;
6090e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int y = yPos;
6100e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    drawText(timeBuff, font, false, &x, &y);
6110e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
6120e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
61339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::parseAnimationDesc(Animation& animation)
614ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood{
615ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    String8 desString;
616ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
61739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (!readFile(animation.zip, "desc.txt", desString)) {
618ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood        return false;
619ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    }
620a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    char const* s = desString.string();
621a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
622a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // Parse the description file
623a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    for (;;) {
624a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const char* endl = strstr(s, "\n");
625cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe        if (endl == NULL) break;
626a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        String8 line(s, endl - s);
627a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const char* l = line.string();
628a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int fps = 0;
629a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int width = 0;
630a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int height = 0;
631a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int count = 0;
632a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int pause = 0;
633afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        char path[ANIM_ENTRY_NAME_MAX];
634083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        char color[7] = "000000"; // default to black if unspecified
6350e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char clockPos1[TEXT_POS_LEN_MAX + 1] = "";
6360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char clockPos2[TEXT_POS_LEN_MAX + 1] = "";
637083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
638d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        char pathType;
639a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        if (sscanf(l, "%d %d %d", &width, &height, &fps) == 3) {
640083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            // ALOGD("> w=%d, h=%d, fps=%d", width, height, fps);
641a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.width = width;
642a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.height = height;
643a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.fps = fps;
6440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        } else if (sscanf(l, " %c %d %d %s #%6s %16s %16s",
6450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                          &pathType, &count, &pause, path, color, clockPos1, clockPos2) >= 4) {
6460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            //ALOGD("> type=%c, count=%d, pause=%d, path=%s, color=%s, clockPos1=%s, clockPos2=%s",
6470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            //    pathType, count, pause, path, color, clockPos1, clockPos2);
648a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            Animation::Part part;
649d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            part.playUntilComplete = pathType == 'c';
650a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.count = count;
651a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.pause = pause;
652a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.path = path;
653d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch            part.audioData = NULL;
65439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.animation = NULL;
655083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            if (!parseColor(color, part.backgroundColor)) {
656083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                ALOGE("> invalid color '#%s'", color);
657083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[0] = 0.0f;
658083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[1] = 0.0f;
659083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[2] = 0.0f;
660083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            }
6610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            parsePosition(clockPos1, clockPos2, &part.clockPosX, &part.clockPosY);
662a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.parts.add(part);
663a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
66439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        else if (strcmp(l, "$SYSTEM") == 0) {
66539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            // ALOGD("> SYSTEM");
66639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            Animation::Part part;
66739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.playUntilComplete = false;
66839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.count = 1;
66939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.pause = 0;
670d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch            part.audioData = NULL;
67139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.animation = loadAnimation(String8(SYSTEM_BOOTANIMATION_FILE));
67239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            if (part.animation != NULL)
67339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy                animation.parts.add(part);
67439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        }
675a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        s = ++endl;
676a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
677a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
67839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
67939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
68039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
68139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::preloadZip(Animation& animation)
68239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
683a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // read all the data structures
684a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const size_t pcount = animation.parts.size();
685afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    void *cookie = NULL;
686dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    ZipFileRO* zip = animation.zip;
687dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    if (!zip->startIteration(&cookie)) {
688afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        return false;
689afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    }
690afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
691a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    Animation::Part* partWithAudio = NULL;
692afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    ZipEntryRO entry;
693afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    char name[ANIM_ENTRY_NAME_MAX];
694dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    while ((entry = zip->nextEntry(cookie)) != NULL) {
695dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        const int foundEntryName = zip->getEntryFileName(entry, name, ANIM_ENTRY_NAME_MAX);
696afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        if (foundEntryName > ANIM_ENTRY_NAME_MAX || foundEntryName == -1) {
697afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            ALOGE("Error fetching entry file name");
698afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            continue;
699afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        }
700afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
701afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 entryName(name);
702afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 path(entryName.getPathDir());
703afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 leaf(entryName.getPathLeaf());
704afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        if (leaf.size() > 0) {
7050e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            if (entryName == CLOCK_FONT_ZIP_NAME) {
7060e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                FileMap* map = zip->createEntryFileMap(entry);
7070e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                if (map) {
7080e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    animation.clockFont.map = map;
7090e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                }
7100e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                continue;
7110e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            }
7120e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
713dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            for (size_t j = 0; j < pcount; j++) {
714afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                if (path == animation.parts[j].path) {
7154600dd053dbdbd4b95f3b11057a1cc55b99f9c77Narayan Kamath                    uint16_t method;
716afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                    // supports only stored png files
717dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                    if (zip->getEntryInfo(entry, &method, NULL, NULL, NULL, NULL, NULL)) {
718afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                        if (method == ZipFileRO::kCompressStored) {
719dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                            FileMap* map = zip->createEntryFileMap(entry);
720afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                            if (map) {
721afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                                Animation::Part& part(animation.parts.editItemAt(j));
722ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                if (leaf == "audio.wav") {
723ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    // a part may have at most one audio file
724d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                                    part.audioData = (uint8_t *)map->getDataPtr();
725d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                                    part.audioLength = map->getDataLength();
726a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch                                    partWithAudio = &part;
727dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                } else if (leaf == "trim.txt") {
728dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    part.trimData.setTo((char const*)map->getDataPtr(),
729dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                                        map->getDataLength());
730ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                } else {
731ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    Animation::Frame frame;
732ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    frame.name = leaf;
733ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    frame.map = map;
734dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimWidth = animation.width;
735dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimHeight = animation.height;
736dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimX = 0;
737dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimY = 0;
738ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    part.frames.add(frame);
739ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                }
740a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                            }
741d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                        } else {
742d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                            ALOGE("bootanimation.zip is compressed; must be only stored");
743a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        }
744a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    }
745a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                }
746a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
747a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
748a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
749a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
750dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    // If there is trimData present, override the positioning defaults.
751dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    for (Animation::Part& part : animation.parts) {
752dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        const char* trimDataStr = part.trimData.string();
753dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        for (size_t frameIdx = 0; frameIdx < part.frames.size(); frameIdx++) {
754dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            const char* endl = strstr(trimDataStr, "\n");
755dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            // No more trimData for this part.
756dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            if (endl == NULL) {
757dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                break;
758dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            }
759dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            String8 line(trimDataStr, endl - trimDataStr);
760dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            const char* lineStr = line.string();
761dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            trimDataStr = ++endl;
762dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            int width = 0, height = 0, x = 0, y = 0;
763dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            if (sscanf(lineStr, "%dx%d+%d+%d", &width, &height, &x, &y) == 4) {
764dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                Animation::Frame& frame(part.frames.editItemAt(frameIdx));
765dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimWidth = width;
766dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimHeight = height;
767dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimX = x;
768dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimY = y;
769dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            } else {
770dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                ALOGE("Error parsing trim.txt, line: %s", lineStr);
771dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                break;
772dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            }
773dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        }
774dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    }
775dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch
776d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch    // Create and initialize audioplay if there is a wav file in any of the animations.
777a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    if (partWithAudio != NULL) {
778d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch        ALOGD("found audio.wav, creating playback engine");
779a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch        if (!audioplay::create(partWithAudio->audioData, partWithAudio->audioLength)) {
780a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch            return false;
781a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch        }
782d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch    }
783d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
784dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    zip->endIteration(cookie);
785afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
78639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
78739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
78839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
78939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::movie()
79039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
79139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    Animation* animation = loadAnimation(mZipFileName);
79239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (animation == NULL)
79339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return false;
79439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
7959748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    bool anyPartHasClock = false;
7969748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    for (size_t i=0; i < animation->parts.size(); i++) {
7970e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if(validClock(animation->parts[i])) {
7989748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            anyPartHasClock = true;
7999748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            break;
8009748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
8019748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8029748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (!anyPartHasClock) {
8039748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mClockEnabled = false;
8049748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8059748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
806271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    // Check if npot textures are supported
807271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    mUseNpotTextures = false;
808271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    String8 gl_extensions;
809271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    const char* exts = reinterpret_cast<const char*>(glGetString(GL_EXTENSIONS));
810271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    if (!exts) {
811271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        glGetError();
812271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    } else {
813271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        gl_extensions.setTo(exts);
814271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        if ((gl_extensions.find("GL_ARB_texture_non_power_of_two") != -1) ||
815271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            (gl_extensions.find("GL_OES_texture_npot") != -1)) {
816271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            mUseNpotTextures = true;
817271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        }
818271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    }
819271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar
820a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    // Blend required to draw time on top of animation frames.
821a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
822a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glShadeModel(GL_FLAT);
823a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_DITHER);
824a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_SCISSOR_TEST);
825a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_BLEND);
826a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
827a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glBindTexture(GL_TEXTURE_2D, 0);
828a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glEnable(GL_TEXTURE_2D);
829a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
830a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
831a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
832a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
833a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
834a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
8350e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    bool clockFontInitialized = false;
836a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    if (mClockEnabled) {
8370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        clockFontInitialized =
8380e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            (initFont(&animation->clockFont, CLOCK_FONT_ASSET) == NO_ERROR);
8390e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        mClockEnabled = clockFontInitialized;
840a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
841a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
8429748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mClockEnabled && !updateIsTimeAccurate()) {
8439748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread = new TimeCheckThread(this);
8449748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread->run("BootAnimation::TimeCheckThread", PRIORITY_NORMAL);
8459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8469748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
84739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    playAnimation(*animation);
8489748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
8499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mTimeCheckThread != NULL) {
8509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread->requestExit();
8519748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread = NULL;
8529748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8539748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
85439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    releaseAnimation(animation);
85539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
8560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (clockFontInitialized) {
8570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glDeleteTextures(1, &animation->clockFont.texture.name);
858815e51d76cc284f3ca59534d9f9372d3dd50f574Andriy Naborskyy    }
859815e51d76cc284f3ca59534d9f9372d3dd50f574Andriy Naborskyy
86039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return false;
86139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
86239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
86339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::playAnimation(const Animation& animation)
86439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
86539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    const size_t pcount = animation.parts.size();
866a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    nsecs_t frameDuration = s2ns(1) / animation.fps;
867dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    const int animationX = (mWidth - animation.width) / 2;
868dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    const int animationY = (mHeight - animation.height) / 2;
8699f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian
870afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    for (size_t i=0 ; i<pcount ; i++) {
871a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const Animation::Part& part(animation.parts[i]);
872a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const size_t fcount = part.frames.size();
873a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        glBindTexture(GL_TEXTURE_2D, 0);
874a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
87539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        // Handle animation package
87639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        if (part.animation != NULL) {
87739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            playAnimation(*part.animation);
87839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            if (exitPending())
87939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy                break;
88039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            continue; //to next part
88139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        }
88239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
883a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        for (int r=0 ; !part.count || r<part.count ; r++) {
884d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            // Exit any non playuntil complete parts immediately
885d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            if(exitPending() && !part.playUntilComplete)
886d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                break;
887d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
888ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood            // only play audio file the first time we animate the part
889290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch            if (r == 0 && part.audioData && playSoundsAllowed()) {
890290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch                ALOGD("playing clip for part%d, size=%d", (int) i, part.audioLength);
891290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch                audioplay::playClip(part.audioData, part.audioLength);
892ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood            }
893ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
894083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            glClearColor(
895083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[0],
896083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[1],
897083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[2],
898083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    1.0f);
899083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
900afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            for (size_t j=0 ; j<fcount && (!exitPending() || part.playUntilComplete) ; j++) {
901a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                const Animation::Frame& frame(part.frames[j]);
902db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                nsecs_t lastFrame = systemTime();
903a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
904a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                if (r > 0) {
905a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    glBindTexture(GL_TEXTURE_2D, frame.tid);
906a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                } else {
907a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    if (part.count != 1) {
908a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glGenTextures(1, &frame.tid);
909a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glBindTexture(GL_TEXTURE_2D, frame.tid);
910a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
911a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
912a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    }
9130e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    int w, h;
9140e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    initTexture(frame.map, &w, &h);
915a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                }
916a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
917dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                const int xc = animationX + frame.trimX;
918dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                const int yc = animationY + frame.trimY;
919dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                Region clearReg(Rect(mWidth, mHeight));
920dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                clearReg.subtractSelf(Rect(xc, yc, xc+frame.trimWidth, yc+frame.trimHeight));
9219f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                if (!clearReg.isEmpty()) {
9229f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    Region::const_iterator head(clearReg.begin());
9239f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    Region::const_iterator tail(clearReg.end());
9249f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    glEnable(GL_SCISSOR_TEST);
9259f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    while (head != tail) {
926cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe                        const Rect& r2(*head++);
927dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                        glScissor(r2.left, mHeight - r2.bottom, r2.width(), r2.height());
9289f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                        glClear(GL_COLOR_BUFFER_BIT);
9299f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    }
9309f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    glDisable(GL_SCISSOR_TEST);
9319f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                }
932dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                // specify the y center as ceiling((mHeight - frame.trimHeight) / 2)
933dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                // which is equivalent to mHeight - (yc + frame.trimHeight)
934dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                glDrawTexiOES(xc, mHeight - (yc + frame.trimHeight),
935dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                              0, frame.trimWidth, frame.trimHeight);
9360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                if (mClockEnabled && mTimeIsAccurate && validClock(part)) {
9370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    drawClock(animation.clockFont, part.clockPosX, part.clockPosY);
938a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi                }
939a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
940a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                eglSwapBuffers(mDisplay, mSurface);
941a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
942a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                nsecs_t now = systemTime();
943a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                nsecs_t delay = frameDuration - (now - lastFrame);
944db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                //ALOGD("%lld, %lld", ns2ms(now - lastFrame), ns2ms(delay));
945a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                lastFrame = now;
946db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian
947db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                if (delay > 0) {
948db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    struct timespec spec;
949db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    spec.tv_sec  = (now + delay) / 1000000000;
950db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    spec.tv_nsec = (now + delay) % 1000000000;
951db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    int err;
952db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    do {
953db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                        err = clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &spec, NULL);
954db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    } while (err<0 && errno == EINTR);
955db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                }
956d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
957d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                checkExit();
958a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
959d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
960a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            usleep(part.pause * ns2us(frameDuration));
961d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
962d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            // For infinite parts, we've now played them at least once, so perhaps exit
963d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            if(exitPending() && !part.count)
964d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                break;
965a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
966a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
9672fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    }
9682fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch
9692fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    // Free textures created for looping parts now that the animation is done.
9702fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    for (const Animation::Part& part : animation.parts) {
971a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        if (part.count != 1) {
9722fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch            const size_t fcount = part.frames.size();
9732fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch            for (size_t j = 0; j < fcount; j++) {
974a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                const Animation::Frame& frame(part.frames[j]);
975a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glDeleteTextures(1, &frame.tid);
976a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
977a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
978a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
979d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
980d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch    // we've finally played everything we're going to play
981d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch    audioplay::setPlaying(false);
982d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch    audioplay::destroy();
983d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
98439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
98539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
986a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
98739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyyvoid BootAnimation::releaseAnimation(Animation* animation) const
98839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
98939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    for (Vector<Animation::Part>::iterator it = animation->parts.begin(),
99039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy         e = animation->parts.end(); it != e; ++it) {
99139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        if (it->animation)
99239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            releaseAnimation(it->animation);
99339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
99439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (animation->zip)
99539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        delete animation->zip;
99639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    delete animation;
997a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian}
998a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
99939218ba26d5bb8646f04273f2c3731598721c1daAndriy NaborskyyBootAnimation::Animation* BootAnimation::loadAnimation(const String8& fn)
100039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
100139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (mLoadedFiles.indexOf(fn) >= 0) {
100239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        ALOGE("File \"%s\" is already loaded. Cyclic ref is not allowed",
100339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            fn.string());
100439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return NULL;
100539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
100639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    ZipFileRO *zip = ZipFileRO::open(fn);
100739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (zip == NULL) {
100839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        ALOGE("Failed to open animation zip \"%s\": %s",
100939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            fn.string(), strerror(errno));
101039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return NULL;
1011a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
1012a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
101339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    Animation *animation =  new Animation;
101439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    animation->fileName = fn;
101539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    animation->zip = zip;
10160e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    animation->clockFont.map = nullptr;
101739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    mLoadedFiles.add(animation->fileName);
101839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
101939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    parseAnimationDesc(*animation);
1020a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    if (!preloadZip(*animation)) {
1021a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch        return NULL;
1022a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    }
1023a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch
1024a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
102539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    mLoadedFiles.remove(fn);
102639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return animation;
102739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
10289748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
1029290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitschbool BootAnimation::playSoundsAllowed() const {
1030290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    // Only play sounds for system boots, not runtime restarts.
1031290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    if (!mSystemBoot) {
1032290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch        return false;
1033290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    }
1034290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch
1035290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    // Read the system property to see if we should play the sound.
1036290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    // If it's not present, default to allowed.
1037290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    if (!property_get_bool(PLAY_SOUND_PROP_NAME, 1)) {
1038290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch        return false;
1039290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    }
1040290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch
1041290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    // Don't play sounds if this is a reboot due to an error.
1042290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    char bootreason[PROPERTY_VALUE_MAX];
1043290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    if (property_get(BOOTREASON_PROP_NAME, bootreason, nullptr) > 0) {
1044290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch        for (const auto& str : PLAY_SOUND_BOOTREASON_BLACKLIST) {
1045290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch            if (strcasecmp(str.c_str(), bootreason) == 0) {
1046290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch                return false;
1047290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch            }
1048290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch        }
1049290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    }
1050290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch    return true;
1051290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch}
1052290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch
10539748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::updateIsTimeAccurate() {
10549748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr long long MAX_TIME_IN_PAST =   60000LL * 60LL * 24LL * 30LL;  // 30 days
10559748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr long long MAX_TIME_IN_FUTURE = 60000LL * 90LL;  // 90 minutes
10569748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10579748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mTimeIsAccurate) {
10589748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10599748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10609748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10619748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    struct stat statResult;
10629748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if(stat(ACCURATE_TIME_FLAG_FILE_PATH, &statResult) == 0) {
10639748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeIsAccurate = true;
10649748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10659748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10669748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10679748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    FILE* file = fopen(LAST_TIME_CHANGED_FILE_PATH, "r");
10689748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (file != NULL) {
10699748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      long long lastChangedTime = 0;
10709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      fscanf(file, "%lld", &lastChangedTime);
10719748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      fclose(file);
10729748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      if (lastChangedTime > 0) {
10739748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        struct timespec now;
10749748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        clock_gettime(CLOCK_REALTIME, &now);
10759748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        // Match the Java timestamp format
10769748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        long long rtcNow = (now.tv_sec * 1000LL) + (now.tv_nsec / 1000000LL);
10779676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi        if (ACCURATE_TIME_EPOCH < rtcNow
10789676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi            && lastChangedTime > (rtcNow - MAX_TIME_IN_PAST)
10799676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi            && lastChangedTime < (rtcNow + MAX_TIME_IN_FUTURE)) {
10809748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            mTimeIsAccurate = true;
10819748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
10829748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      }
10839748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10849748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10859748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return mTimeIsAccurate;
10869748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10879748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10889748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::TimeCheckThread::TimeCheckThread(BootAnimation* bootAnimation) : Thread(false),
10899748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mInotifyFd(-1), mSystemWd(-1), mTimeWd(-1), mBootAnimation(bootAnimation) {}
10909748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10919748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::TimeCheckThread::~TimeCheckThread() {
10929748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    // mInotifyFd may be -1 but that's ok since we're not at risk of attempting to close a valid FD.
10939748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    close(mInotifyFd);
10949748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10959748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10969748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::TimeCheckThread::threadLoop() {
10979748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    bool shouldLoop = doThreadLoop() && !mBootAnimation->mTimeIsAccurate
10989748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        && mBootAnimation->mClockEnabled;
10999748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (!shouldLoop) {
11009748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11019748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11029748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11039748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return shouldLoop;
11049748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11059748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11069748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::TimeCheckThread::doThreadLoop() {
11079748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr int BUFF_LEN (10 * (sizeof(struct inotify_event) + NAME_MAX + 1));
11089748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11099748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    // Poll instead of doing a blocking read so the Thread can exit if requested.
11109748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    struct pollfd pfd = { mInotifyFd, POLLIN, 0 };
11119748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    ssize_t pollResult = poll(&pfd, 1, 1000);
11129748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11139748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (pollResult == 0) {
11149748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
11159748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    } else if (pollResult < 0) {
11169748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not poll inotify events");
11179748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return false;
11189748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11199748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11209748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    char buff[BUFF_LEN] __attribute__ ((aligned(__alignof__(struct inotify_event))));;
11219748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    ssize_t length = read(mInotifyFd, buff, BUFF_LEN);
11229748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (length == 0) {
11239748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
11249748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    } else if (length < 0) {
11259748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not read inotify events");
11269748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return false;
11279748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11289748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11299748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    const struct inotify_event *event;
11309748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    for (char* ptr = buff; ptr < buff + length; ptr += sizeof(struct inotify_event) + event->len) {
11319748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        event = (const struct inotify_event *) ptr;
11329748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        if (event->wd == mSystemWd && strcmp(SYSTEM_TIME_DIR_NAME, event->name) == 0) {
11339748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            addTimeDirWatch();
11349748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        } else if (event->wd == mTimeWd && (strcmp(LAST_TIME_CHANGED_FILE_NAME, event->name) == 0
11359748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi                || strcmp(ACCURATE_TIME_FLAG_FILE_NAME, event->name) == 0)) {
11369748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            return !mBootAnimation->updateIsTimeAccurate();
11379748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
11389748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11399748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11409748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return true;
11419748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11429748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11439748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchivoid BootAnimation::TimeCheckThread::addTimeDirWatch() {
11449748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeWd = inotify_add_watch(mInotifyFd, SYSTEM_TIME_DIR_PATH,
11459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi                IN_CLOSE_WRITE | IN_MOVED_TO | IN_ATTRIB);
11469748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        if (mTimeWd > 0) {
11479748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            // No need to watch for the time directory to be created if it already exists
11489748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            inotify_rm_watch(mInotifyFd, mSystemWd);
11499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            mSystemWd = -1;
11509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
11519748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11529748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11539748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatus_t BootAnimation::TimeCheckThread::readyToRun() {
11549748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mInotifyFd = inotify_init();
11559748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mInotifyFd < 0) {
11569748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not initialize inotify fd");
11579748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return NO_INIT;
11589748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11599748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11609748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mSystemWd = inotify_add_watch(mInotifyFd, SYSTEM_DATA_DIR_PATH, IN_CREATE | IN_ATTRIB);
11619748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mSystemWd < 0) {
11629748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11639748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11649748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not add watch for %s", SYSTEM_DATA_DIR_PATH);
11659748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return NO_INIT;
11669748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11679748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11689748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    addTimeDirWatch();
11699748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mBootAnimation->updateIsTimeAccurate()) {
11719748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11729748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11739748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return ALREADY_EXISTS;
11749748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11759748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11769748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return NO_ERROR;
11779748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11789748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project; // namespace android
1183