BootAnimation.cpp revision e13b58e15b561b1b85788800e0c3af48fa80463e
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>
21a90c54c90e1cb668cd74f218370f868763bbd009Wei Wang#include <inttypes.h>
229748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi#include <sys/inotify.h>
239748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi#include <sys/poll.h>
249748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi#include <sys/stat.h>
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/types.h>
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <math.h>
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <fcntl.h>
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/misc.h>
29b4d5a72691846bce5779cc3db056f09a5031d7ccMathias Agopian#include <signal.h>
30bb94f3107ea567ddf67bed4617c3bcbd602538dfElliott Hughes#include <time.h>
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
32bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks#include <cutils/properties.h>
33bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
34b13b9bdad2baf6ad1ec2e56b6b7598fa20f55fc4Mathias Agopian#include <androidfw/AssetManager.h>
35ac31a3b8b09aba1c5ebc73f0cf65cac2210aa6b7Mathias Agopian#include <binder/IPCThreadState.h>
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Atomic.h>
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Errors.h>
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Log.h>
39a90c54c90e1cb668cd74f218370f868763bbd009Wei Wang#include <utils/SystemClock.h>
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
41b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park#include <android-base/properties.h>
42b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/PixelFormat.h>
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/Rect.h>
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/Region.h>
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/DisplayInfo.h>
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
480b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown#include <gui/ISurfaceComposer.h>
498335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian#include <gui/Surface.h>
508335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian#include <gui/SurfaceComposerClient.h>
51000479f9e325b4e426a67033abd92d47da412725Mathias Agopian
52cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe// TODO: Fix Skia.
53cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe#pragma GCC diagnostic push
54cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe#pragma GCC diagnostic ignored "-Wunused-parameter"
55eece0dda56ae29fff6e9003df97594f6ac50b6e2Derek Sollenberger#include <SkBitmap.h>
568898c16987831db47542a92151ed7209e22268aaMatt Sarett#include <SkImage.h>
57eece0dda56ae29fff6e9003df97594f6ac50b6e2Derek Sollenberger#include <SkStream.h>
58cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe#pragma GCC diagnostic pop
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/gl.h>
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/glext.h>
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <EGL/eglext.h>
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "BootAnimation.h"
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
689748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char OEM_BOOTANIMATION_FILE[] = "/oem/media/bootanimation.zip";
699748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_BOOTANIMATION_FILE[] = "/system/media/bootanimation.zip";
709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_ENCRYPTED_BOOTANIMATION_FILE[] = "/system/media/bootanimation-encrypted.zip";
71b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Parkstatic const char OEM_SHUTDOWNANIMATION_FILE[] = "/oem/media/shutdownanimation.zip";
72b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Parkstatic const char SYSTEM_SHUTDOWNANIMATION_FILE[] = "/system/media/shutdownanimation.zip";
73b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park
749748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_DATA_DIR_PATH[] = "/data/system";
759748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_TIME_DIR_NAME[] = "time";
769748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_TIME_DIR_PATH[] = "/data/system/time";
770e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const char CLOCK_FONT_ASSET[] = "images/clock_font.png";
780e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const char CLOCK_FONT_ZIP_NAME[] = "clock_font.png";
799748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char LAST_TIME_CHANGED_FILE_NAME[] = "last_time_change";
809748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char LAST_TIME_CHANGED_FILE_PATH[] = "/data/system/time/last_time_change";
819748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char ACCURATE_TIME_FLAG_FILE_NAME[] = "time_is_accurate";
829748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char ACCURATE_TIME_FLAG_FILE_PATH[] = "/data/system/time/time_is_accurate";
839071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchistatic const char TIME_FORMAT_12_HOUR_FLAG_FILE_PATH[] = "/data/system/time/time_format_12_hour";
849676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi// Java timestamp format. Don't show the clock if the date is before 2000-01-01 00:00:00.
859676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchistatic const long long ACCURATE_TIME_EPOCH = 946684800000;
860e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr char FONT_BEGIN_CHAR = ' ';
870e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr char FONT_END_CHAR = '~' + 1;
880e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_CHARS = FONT_END_CHAR - FONT_BEGIN_CHAR + 1;
890e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_COLS = 16;
900e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_ROWS = FONT_NUM_CHARS / FONT_NUM_COLS;
910e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const int TEXT_CENTER_VALUE = INT_MAX;
920e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const int TEXT_MISSING_VALUE = INT_MIN;
939748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char EXIT_PROP_NAME[] = "service.bootanim.exit";
94afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamathstatic const int ANIM_ENTRY_NAME_MAX = 256;
950e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t TEXT_POS_LEN_MAX = 16;
96afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
997464ac9bd7fe89061e47617e4b6004b88c91d636Ed CoyneBootAnimation::BootAnimation(sp<Callbacks> callbacks)
1002c9e94aa3e9614f0d202ee008c08a9011fa7f75fEd Coyne        : Thread(false), mClockEnabled(true), mTimeIsAccurate(false),
1017464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne        mTimeFormat12Hour(false), mTimeCheckThread(NULL), mCallbacks(callbacks) {
102627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    mSession = new SurfaceComposerClient();
103290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch
104b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    std::string powerCtl = android::base::GetProperty("sys.powerctl", "");
105b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    if (powerCtl.empty()) {
106b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        mShuttingDown = false;
107b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    } else {
108b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        mShuttingDown = true;
109b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    }
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid BootAnimation::onFirstRef() {
113bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    status_t err = mSession->linkToComposerDeath(this);
1143762c311729fe9f3af085c14c5c1fb471d994c03Steve Block    ALOGE_IF(err, "linkToComposerDeath failed (%s) ", strerror(-err));
1158434c5369304e639efe8eab368ca410c589d87c2Mathias Agopian    if (err == NO_ERROR) {
116bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian        run("BootAnimation", PRIORITY_DISPLAY);
117bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    }
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
120bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopiansp<SurfaceComposerClient> BootAnimation::session() const {
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mSession;
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
124bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
125afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamathvoid BootAnimation::binderDied(const wp<IBinder>&)
126bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian{
127bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // woah, surfaceflinger died!
1285baa3a62a97544669fba6d65a11c07f252e654ddSteve Block    ALOGD("SurfaceFlinger died, exiting...");
129bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
130bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // calling requestExit() is not enough here because the Surface code
131bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // might be blocked on a condition variable that will never be updated.
132bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    kill( getpid(), SIGKILL );
133bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    requestExit();
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    const int w = bitmap.width();
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int h = bitmap.height();
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const void* p = bitmap.getPixels();
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    GLint crop[4] = { 0, h, w, -h };
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    texture->w = w;
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    texture->h = h;
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &texture->name);
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, texture->name);
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
16042a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed    switch (bitmap.colorType()) {
16142a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kAlpha_8_SkColorType:
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, w, h, 0, GL_ALPHA,
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_BYTE, p);
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
16542a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kARGB_4444_SkColorType:
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_SHORT_4_4_4_4, p);
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
16942a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kN32_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_BYTE, p);
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
17342a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kRGB_565_SkColorType:
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB,
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_SHORT_5_6_5, p);
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        default:
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, crop);
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
1860e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1900e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatus_t BootAnimation::initTexture(FileMap* map, int* width, int* height)
191a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
192a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    SkBitmap bitmap;
193f67b317c102db72182046ec2a2cffe2331017709Damien Bargiacchi    sk_sp<SkData> data = SkData::MakeWithoutCopy(map->getDataPtr(),
194f67b317c102db72182046ec2a2cffe2331017709Damien Bargiacchi            map->getDataLength());
1958898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkImage> image = SkImage::MakeFromEncoded(data);
1968898c16987831db47542a92151ed7209e22268aaMatt Sarett    image->asLegacyBitmap(&bitmap, SkImage::kRO_LegacyBitmapMode);
197a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
1980c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // FileMap memory is never released until application exit.
1990c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // Release it now as the texture is already loaded and the memory used for
2000c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // the packed resource can be released.
2010e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    delete map;
2020c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko
203a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const int w = bitmap.width();
204a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const int h = bitmap.height();
205a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const void* p = bitmap.getPixels();
206a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
207a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    GLint crop[4] = { 0, h, w, -h };
208a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    int tw = 1 << (31 - __builtin_clz(w));
209a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    int th = 1 << (31 - __builtin_clz(h));
210a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    if (tw < w) tw <<= 1;
211a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    if (th < h) th <<= 1;
212a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
21342a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed    switch (bitmap.colorType()) {
21442a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kN32_SkColorType:
215271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            if (!mUseNpotTextures && (tw != w || th != h)) {
216a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, tw, th, 0, GL_RGBA,
217a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_BYTE, 0);
218a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexSubImage2D(GL_TEXTURE_2D, 0,
219a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        0, 0, w, h, GL_RGBA, GL_UNSIGNED_BYTE, p);
220a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            } else {
221271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
222a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_BYTE, p);
223a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
224a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
225a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
22642a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kRGB_565_SkColorType:
227271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            if (!mUseNpotTextures && (tw != w || th != h)) {
228a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, tw, th, 0, GL_RGB,
229a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_SHORT_5_6_5, 0);
230a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexSubImage2D(GL_TEXTURE_2D, 0,
231a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        0, 0, w, h, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, p);
232a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            } else {
233271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB,
234a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_SHORT_5_6_5, p);
235a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
236a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
237a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        default:
238a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
239a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
240a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
241a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, crop);
242a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *width = w;
2440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *height = h;
2450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
246a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    return NO_ERROR;
247a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian}
248a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t BootAnimation::readyToRun() {
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mAssets.addDefaultAssets();
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2520b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    sp<IBinder> dtoken(SurfaceComposerClient::getBuiltInDisplay(
2530b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown            ISurfaceComposer::eDisplayIdMain));
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    DisplayInfo dinfo;
2550b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    status_t status = SurfaceComposerClient::getDisplayInfo(dtoken, &dinfo);
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (status)
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return -1;
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // create the native surface
2600b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    sp<SurfaceControl> control = session()->createSurface(String8("BootAnimation"),
2610b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown            dinfo.w, dinfo.h, PIXEL_FORMAT_RGB_565);
262439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian
263e13b58e15b561b1b85788800e0c3af48fa80463eRobert Carr    SurfaceComposerClient::Transaction t;
264e13b58e15b561b1b85788800e0c3af48fa80463eRobert Carr    t.setLayer(control, 0x40000000)
265e13b58e15b561b1b85788800e0c3af48fa80463eRobert Carr        .apply();
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26717f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    sp<Surface> s = control->getSurface();
26817f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize opengl and egl
270738b9a45399485893fa9541fae19c47e01bc4af4Mathias Agopian    const EGLint attribs[] = {
2711b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_RED_SIZE,   8,
2721b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_GREEN_SIZE, 8,
2731b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_BLUE_SIZE,  8,
274a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            EGL_DEPTH_SIZE, 0,
275a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            EGL_NONE
276738b9a45399485893fa9541fae19c47e01bc4af4Mathias Agopian    };
277cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe    EGLint w, h;
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLint numConfigs;
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLConfig config;
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLSurface surface;
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLContext context;
282627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
284627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
285627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    eglInitialize(display, 0, 0);
2861b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian    eglChooseConfig(display, attribs, &config, 1, &numConfigs);
2871473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    surface = eglCreateWindowSurface(display, config, s.get(), NULL);
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    context = eglCreateContext(display, config, NULL, NULL);
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglQuerySurface(display, surface, EGL_WIDTH, &w);
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglQuerySurface(display, surface, EGL_HEIGHT, &h);
291a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
292abac01052c1c40f8582fe9f010c2efe6013e25c6Mathias Agopian    if (eglMakeCurrent(display, surface, surface, context) == EGL_FALSE)
293abac01052c1c40f8582fe9f010c2efe6013e25c6Mathias Agopian        return NO_INIT;
294a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDisplay = display;
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mContext = context;
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSurface = surface;
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mWidth = w;
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mHeight = h;
30017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    mFlingerSurfaceControl = control;
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFlingerSurface = s;
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
303c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes    // If the device has encryption turned on or is in process
304bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    // of being encrypted we show the encrypted boot animation.
305bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    char decrypt[PROPERTY_VALUE_MAX];
306bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    property_get("vold.decrypt", decrypt, "");
307bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
308b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    bool encryptedAnimation = atoi(decrypt) != 0 ||
309b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        !strcmp("trigger_restart_min_framework", decrypt);
310bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
311b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    if (!mShuttingDown && encryptedAnimation &&
312b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        (access(SYSTEM_ENCRYPTED_BOOTANIMATION_FILE, R_OK) == 0)) {
31339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        mZipFileName = SYSTEM_ENCRYPTED_BOOTANIMATION_FILE;
314b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        return NO_ERROR;
31539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
316b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    static const char* bootFiles[] = {OEM_BOOTANIMATION_FILE, SYSTEM_BOOTANIMATION_FILE};
317b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    static const char* shutdownFiles[] =
318b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        {OEM_SHUTDOWNANIMATION_FILE, SYSTEM_SHUTDOWNANIMATION_FILE};
319b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park
320b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    for (const char* f : (!mShuttingDown ? bootFiles : shutdownFiles)) {
321b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        if (access(f, R_OK) == 0) {
322b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park            mZipFileName = f;
323b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park            return NO_ERROR;
324b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        }
325bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    }
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
329a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopianbool BootAnimation::threadLoop()
330a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
331a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    bool r;
332afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    // We have no bootanimation file, so we use the stock android logo
333afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    // animation.
33439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (mZipFileName.isEmpty()) {
335a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        r = android();
336a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    } else {
337a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        r = movie();
338a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
339a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglMakeCurrent(mDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglDestroyContext(mDisplay, mContext);
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglDestroySurface(mDisplay, mSurface);
3436cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    mFlingerSurface.clear();
34417f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    mFlingerSurfaceControl.clear();
345627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    eglTerminate(mDisplay);
3463eee9fbb6df1dbbcefe1f0dfdf1295c95ddf3279Sai Kiran Korwar    eglReleaseThread();
347627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    IPCThreadState::self()->stopProcess();
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return r;
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
351a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopianbool BootAnimation::android()
352a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
353d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park    ALOGD("%sAnimationShownTiming start time: %" PRId64 "ms", mShuttingDown ? "Shutdown" : "Boot",
354d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park            elapsedRealtime());
355b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    initTexture(&mAndroid[0], mAssets, "images/android-logo-mask.png");
356b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    initTexture(&mAndroid[1], mAssets, "images/android-logo-shine.png");
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3587464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne    mCallbacks->init({});
3597464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear screen
361a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glShadeModel(GL_FLAT);
362b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glDisable(GL_DITHER);
363b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glDisable(GL_SCISSOR_TEST);
36459f19e48c1c043bb9debdc35d166e397e2125d33Mathias Agopian    glClearColor(0,0,0,1);
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glClear(GL_COLOR_BUFFER_BIT);
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglSwapBuffers(mDisplay, mSurface);
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
368a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glEnable(GL_TEXTURE_2D);
369a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
370a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
371b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const GLint xc = (mWidth  - mAndroid[0].w) / 2;
372b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const GLint yc = (mHeight - mAndroid[0].h) / 2;
373b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const Rect updateRect(xc, yc, xc + mAndroid[0].w, yc + mAndroid[0].h);
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glScissor(updateRect.left, mHeight - updateRect.bottom, updateRect.width(),
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            updateRect.height());
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
378b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    // Blend state
379b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
380b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
381b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t startTime = systemTime();
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    do {
3841379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        nsecs_t now = systemTime();
3851379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        double time = now - startTime;
386b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        float t = 4.0f * float(time / us2ns(16667)) / mAndroid[1].w;
387b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        GLint offset = (1 - (t - floorf(t))) * mAndroid[1].w;
388b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        GLint x = xc - offset;
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3908166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glDisable(GL_SCISSOR_TEST);
3918166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
3928166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian
3938166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glEnable(GL_SCISSOR_TEST);
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisable(GL_BLEND);
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mAndroid[1].name);
396b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(x,                 yc, 0, mAndroid[1].w, mAndroid[1].h);
397b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(x + mAndroid[1].w, yc, 0, mAndroid[1].w, mAndroid[1].h);
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
399b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glEnable(GL_BLEND);
400b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glBindTexture(GL_TEXTURE_2D, mAndroid[0].name);
401b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(xc, yc, 0, mAndroid[0].w, mAndroid[0].h);
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
403627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian        EGLBoolean res = eglSwapBuffers(mDisplay, mSurface);
404627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian        if (res == EGL_FALSE)
405627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            break;
406627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
4071379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        // 12fps: don't animate too fast to preserve CPU
4081379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        const nsecs_t sleepTime = 83333 - ns2us(systemTime() - now);
4091379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        if (sleepTime > 0)
410a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            usleep(sleepTime);
411d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
412d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        checkExit();
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } while (!exitPending());
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mAndroid[0].name);
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mAndroid[1].name);
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return false;
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
420d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hestervoid BootAnimation::checkExit() {
421d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    // Allow surface flinger to gracefully request shutdown
422d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    char value[PROPERTY_VALUE_MAX];
423d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    property_get(EXIT_PROP_NAME, value, "0");
424d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    int exitnow = atoi(value);
425d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    if (exitnow) {
426d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        requestExit();
4277464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne        mCallbacks->shutdown();
428d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    }
429d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester}
430d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
4310e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchibool BootAnimation::validClock(const Animation::Part& part) {
4320e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return part.clockPosX != TEXT_MISSING_VALUE && part.clockPosY != TEXT_MISSING_VALUE;
4330e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4340e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4350e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchibool parseTextCoord(const char* str, int* dest) {
4360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (strcmp("c", str) == 0) {
4370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *dest = TEXT_CENTER_VALUE;
4380e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return true;
4390e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4400e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4410e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    char* end;
4420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int val = (int) strtol(str, &end, 0);
4430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (end == str || *end != '\0' || val == INT_MAX || val == INT_MIN) {
4440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return false;
4450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *dest = val;
4470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return true;
4480e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4490e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4500e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// Parse two position coordinates. If only string is non-empty, treat it as the y value.
4510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid parsePosition(const char* str1, const char* str2, int* x, int* y) {
4520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    bool success = false;
4530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (strlen(str1) == 0) {  // No values were specified
4540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        // success = false
4550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (strlen(str2) == 0) {  // we have only one value
4560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (parseTextCoord(str1, y)) {
4570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            *x = TEXT_CENTER_VALUE;
4580e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            success = true;
4590e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        }
4600e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else {
4610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (parseTextCoord(str1, x) && parseTextCoord(str2, y)) {
4620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            success = true;
4630e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        }
4640e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4650e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4660e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (!success) {
4670e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = TEXT_MISSING_VALUE;
4680e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = TEXT_MISSING_VALUE;
4690e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4700e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4710e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
472083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// Parse a color represented as an HTML-style 'RRGGBB' string: each pair of
473083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// characters in str is a hex number in [0, 255], which are converted to
474083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// floating point values in the range [0.0, 1.0] and placed in the
475083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// corresponding elements of color.
476083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall//
477083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// If the input string isn't valid, parseColor returns false and color is
478083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// left unchanged.
479083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hallstatic bool parseColor(const char str[7], float color[3]) {
480083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    float tmpColor[3];
481083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    for (int i = 0; i < 3; i++) {
482083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        int val = 0;
483083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        for (int j = 0; j < 2; j++) {
484083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            val *= 16;
485083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            char c = str[2*i + j];
486083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            if      (c >= '0' && c <= '9') val += c - '0';
487083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else if (c >= 'A' && c <= 'F') val += (c - 'A') + 10;
488083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else if (c >= 'a' && c <= 'f') val += (c - 'a') + 10;
489083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else                           return false;
490083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        }
491083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        tmpColor[i] = static_cast<float>(val) / 255.0f;
492083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    }
493083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    memcpy(color, tmpColor, sizeof(tmpColor));
494083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    return true;
495083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall}
496083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
49739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
49839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyystatic bool readFile(ZipFileRO* zip, const char* name, String8& outString)
499a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
50039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    ZipEntryRO entry = zip->findEntryByName(name);
501ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    ALOGE_IF(!entry, "couldn't find %s", name);
502ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    if (!entry) {
503afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        return false;
504afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    }
505a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
50639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    FileMap* entryMap = zip->createEntryFileMap(entry);
50739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    zip->releaseEntry(entry);
508ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    ALOGE_IF(!entryMap, "entryMap is null");
509ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    if (!entryMap) {
510a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        return false;
511a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
512a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
513ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    outString.setTo((char const*)entryMap->getDataPtr(), entryMap->getDataLength());
514688ff4cf0f93d1a886437884f23874e5595d59a3Narayan Kamath    delete entryMap;
515ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    return true;
516ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood}
517ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
5180e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// The font image should be a 96x2 array of character images.  The
5190e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// columns are the printable ASCII characters 0x20 - 0x7f.  The
5200e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// top row is regular text; the bottom row is bold.
5210e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatus_t BootAnimation::initFont(Font* font, const char* fallback) {
5220e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    status_t status = NO_ERROR;
523a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5240e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (font->map != nullptr) {
5250e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glGenTextures(1, &font->texture.name);
5260e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glBindTexture(GL_TEXTURE_2D, font->texture.name);
527a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5280e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        status = initTexture(font->map, &font->texture.w, &font->texture.h);
5290e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5300e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
5310e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
5320e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
5330e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
5340e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (fallback != nullptr) {
5350e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        status = initTexture(&font->texture, mAssets, fallback);
5360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else {
5370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return NO_INIT;
538a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
539a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5400e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (status == NO_ERROR) {
5410e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        font->char_width = font->texture.w / FONT_NUM_COLS;
5420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        font->char_height = font->texture.h / FONT_NUM_ROWS / 2;  // There are bold and regular rows
5430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
544a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return status;
5460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
547a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5480e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid BootAnimation::drawText(const char* str, const Font& font, bool bold, int* x, int* y) {
5490e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    glEnable(GL_BLEND);  // Allow us to draw on top of the animation
5500e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    glBindTexture(GL_TEXTURE_2D, font.texture.name);
5510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    const int len = strlen(str);
5530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    const int strWidth = font.char_width * len;
5540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (*x == TEXT_CENTER_VALUE) {
5560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = (mWidth - strWidth) / 2;
5570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (*x < 0) {
5580e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = mWidth + *x - strWidth;
5590e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
5600e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (*y == TEXT_CENTER_VALUE) {
5610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = (mHeight - font.char_height) / 2;
5620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (*y < 0) {
5630e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = mHeight + *y - font.char_height;
564a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
565a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5660e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int cropRect[4] = { 0, 0, font.char_width, -font.char_height };
567a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5680e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    for (int i = 0; i < len; i++) {
5690e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char c = str[i];
570a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5710e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (c < FONT_BEGIN_CHAR || c > FONT_END_CHAR) {
5720e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            c = '?';
573a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        }
574a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
575a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        // Crop the texture to only the pixels in the current glyph
5760e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int charPos = (c - FONT_BEGIN_CHAR);  // Position in the list of valid characters
5770e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int row = charPos / FONT_NUM_COLS;
5780e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int col = charPos % FONT_NUM_COLS;
5790e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[0] = col * font.char_width;  // Left of column
5800e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[1] = row * font.char_height * 2; // Top of row
5810e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        // Move down to bottom of regular (one char_heigh) or bold (two char_heigh) line
5820e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[1] += bold ? 2 * font.char_height : font.char_height;
583a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, cropRect);
584a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5850e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glDrawTexiOES(*x, *y, 0, font.char_width, font.char_height);
586a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5870e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x += font.char_width;
588a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
589a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
590a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glDisable(GL_BLEND);  // Return to the animation's default behaviour
591a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glBindTexture(GL_TEXTURE_2D, 0);
592a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi}
593a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5949071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi// We render 12 or 24 hour time.
5950e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid BootAnimation::drawClock(const Font& font, const int xPos, const int yPos) {
5969071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    static constexpr char TIME_FORMAT_12[] = "%l:%M";
5979071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    static constexpr char TIME_FORMAT_24[] = "%H:%M";
5980e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    static constexpr int TIME_LENGTH = 6;
5990e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6000e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    time_t rawtime;
6010e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    time(&rawtime);
6020e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    struct tm* timeInfo = localtime(&rawtime);
6030e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6040e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    char timeBuff[TIME_LENGTH];
6059071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    const char* timeFormat = mTimeFormat12Hour ? TIME_FORMAT_12 : TIME_FORMAT_24;
6069071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    size_t length = strftime(timeBuff, TIME_LENGTH, timeFormat, timeInfo);
6070e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6080e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (length != TIME_LENGTH - 1) {
6090e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        ALOGE("Couldn't format time; abandoning boot animation clock");
6100e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        mClockEnabled = false;
6110e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return;
6120e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
6130e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
61445a764400485fb6fb9b0d3566e24feac446e616cDamien Bargiacchi    char* out = timeBuff[0] == ' ' ? &timeBuff[1] : &timeBuff[0];
6150e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int x = xPos;
6160e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int y = yPos;
61745a764400485fb6fb9b0d3566e24feac446e616cDamien Bargiacchi    drawText(out, font, false, &x, &y);
6180e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
6190e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
62039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::parseAnimationDesc(Animation& animation)
621ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood{
622ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    String8 desString;
623ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
62439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (!readFile(animation.zip, "desc.txt", desString)) {
625ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood        return false;
626ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    }
627a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    char const* s = desString.string();
628a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
629a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // Parse the description file
630a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    for (;;) {
631a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const char* endl = strstr(s, "\n");
632cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe        if (endl == NULL) break;
633a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        String8 line(s, endl - s);
634a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const char* l = line.string();
635a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int fps = 0;
636a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int width = 0;
637a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int height = 0;
638a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int count = 0;
639a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int pause = 0;
640afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        char path[ANIM_ENTRY_NAME_MAX];
641083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        char color[7] = "000000"; // default to black if unspecified
6420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char clockPos1[TEXT_POS_LEN_MAX + 1] = "";
6430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char clockPos2[TEXT_POS_LEN_MAX + 1] = "";
644083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
645d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        char pathType;
646a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        if (sscanf(l, "%d %d %d", &width, &height, &fps) == 3) {
647083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            // ALOGD("> w=%d, h=%d, fps=%d", width, height, fps);
648a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.width = width;
649a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.height = height;
650a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.fps = fps;
6510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        } else if (sscanf(l, " %c %d %d %s #%6s %16s %16s",
6520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                          &pathType, &count, &pause, path, color, clockPos1, clockPos2) >= 4) {
6530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            //ALOGD("> type=%c, count=%d, pause=%d, path=%s, color=%s, clockPos1=%s, clockPos2=%s",
6540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            //    pathType, count, pause, path, color, clockPos1, clockPos2);
655a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            Animation::Part part;
656d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            part.playUntilComplete = pathType == 'c';
657a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.count = count;
658a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.pause = pause;
659a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.path = path;
660d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch            part.audioData = NULL;
66139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.animation = NULL;
662083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            if (!parseColor(color, part.backgroundColor)) {
663083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                ALOGE("> invalid color '#%s'", color);
664083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[0] = 0.0f;
665083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[1] = 0.0f;
666083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[2] = 0.0f;
667083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            }
6680e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            parsePosition(clockPos1, clockPos2, &part.clockPosX, &part.clockPosY);
669a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.parts.add(part);
670a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
67139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        else if (strcmp(l, "$SYSTEM") == 0) {
67239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            // ALOGD("> SYSTEM");
67339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            Animation::Part part;
67439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.playUntilComplete = false;
67539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.count = 1;
67639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.pause = 0;
677d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch            part.audioData = NULL;
67839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.animation = loadAnimation(String8(SYSTEM_BOOTANIMATION_FILE));
67939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            if (part.animation != NULL)
68039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy                animation.parts.add(part);
68139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        }
682a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        s = ++endl;
683a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
684a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
68539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
68639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
68739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
68839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::preloadZip(Animation& animation)
68939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
690a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // read all the data structures
691a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const size_t pcount = animation.parts.size();
692afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    void *cookie = NULL;
693dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    ZipFileRO* zip = animation.zip;
694dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    if (!zip->startIteration(&cookie)) {
695afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        return false;
696afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    }
697afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
698afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    ZipEntryRO entry;
699afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    char name[ANIM_ENTRY_NAME_MAX];
700dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    while ((entry = zip->nextEntry(cookie)) != NULL) {
701dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        const int foundEntryName = zip->getEntryFileName(entry, name, ANIM_ENTRY_NAME_MAX);
702afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        if (foundEntryName > ANIM_ENTRY_NAME_MAX || foundEntryName == -1) {
703afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            ALOGE("Error fetching entry file name");
704afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            continue;
705afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        }
706afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
707afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 entryName(name);
708afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 path(entryName.getPathDir());
709afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 leaf(entryName.getPathLeaf());
710afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        if (leaf.size() > 0) {
7110e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            if (entryName == CLOCK_FONT_ZIP_NAME) {
7120e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                FileMap* map = zip->createEntryFileMap(entry);
7130e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                if (map) {
7140e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    animation.clockFont.map = map;
7150e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                }
7160e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                continue;
7170e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            }
7180e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
719dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            for (size_t j = 0; j < pcount; j++) {
720afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                if (path == animation.parts[j].path) {
7214600dd053dbdbd4b95f3b11057a1cc55b99f9c77Narayan Kamath                    uint16_t method;
722afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                    // supports only stored png files
723dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                    if (zip->getEntryInfo(entry, &method, NULL, NULL, NULL, NULL, NULL)) {
724afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                        if (method == ZipFileRO::kCompressStored) {
725dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                            FileMap* map = zip->createEntryFileMap(entry);
726afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                            if (map) {
727afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                                Animation::Part& part(animation.parts.editItemAt(j));
728ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                if (leaf == "audio.wav") {
729ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    // a part may have at most one audio file
730d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                                    part.audioData = (uint8_t *)map->getDataPtr();
731d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                                    part.audioLength = map->getDataLength();
732dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                } else if (leaf == "trim.txt") {
733dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    part.trimData.setTo((char const*)map->getDataPtr(),
734dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                                        map->getDataLength());
735ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                } else {
736ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    Animation::Frame frame;
737ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    frame.name = leaf;
738ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    frame.map = map;
739dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimWidth = animation.width;
740dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimHeight = animation.height;
741dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimX = 0;
742dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimY = 0;
743ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    part.frames.add(frame);
744ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                }
745a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                            }
746d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                        } else {
747d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                            ALOGE("bootanimation.zip is compressed; must be only stored");
748a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        }
749a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    }
750a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                }
751a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
752a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
753a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
754a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
755dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    // If there is trimData present, override the positioning defaults.
756dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    for (Animation::Part& part : animation.parts) {
757dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        const char* trimDataStr = part.trimData.string();
758dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        for (size_t frameIdx = 0; frameIdx < part.frames.size(); frameIdx++) {
759dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            const char* endl = strstr(trimDataStr, "\n");
760dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            // No more trimData for this part.
761dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            if (endl == NULL) {
762dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                break;
763dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            }
764dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            String8 line(trimDataStr, endl - trimDataStr);
765dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            const char* lineStr = line.string();
766dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            trimDataStr = ++endl;
767dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            int width = 0, height = 0, x = 0, y = 0;
768dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            if (sscanf(lineStr, "%dx%d+%d+%d", &width, &height, &x, &y) == 4) {
769dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                Animation::Frame& frame(part.frames.editItemAt(frameIdx));
770dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimWidth = width;
771dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimHeight = height;
772dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimX = x;
773dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimY = y;
774dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            } else {
775dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                ALOGE("Error parsing trim.txt, line: %s", lineStr);
776dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                break;
777dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            }
778dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        }
779dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    }
780dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch
7817464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne    mCallbacks->init(animation.parts);
782d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
783dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    zip->endIteration(cookie);
784afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
78539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
78639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
78739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
78839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::movie()
78939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
79039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    Animation* animation = loadAnimation(mZipFileName);
79139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (animation == NULL)
79239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return false;
79339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
7949748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    bool anyPartHasClock = false;
7959748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    for (size_t i=0; i < animation->parts.size(); i++) {
7960e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if(validClock(animation->parts[i])) {
7979748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            anyPartHasClock = true;
7989748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            break;
7999748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
8009748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8019748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (!anyPartHasClock) {
8029748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mClockEnabled = false;
8039748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8049748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
805271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    // Check if npot textures are supported
806271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    mUseNpotTextures = false;
807271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    String8 gl_extensions;
808271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    const char* exts = reinterpret_cast<const char*>(glGetString(GL_EXTENSIONS));
809271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    if (!exts) {
810271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        glGetError();
811271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    } else {
812271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        gl_extensions.setTo(exts);
813271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        if ((gl_extensions.find("GL_ARB_texture_non_power_of_two") != -1) ||
814271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            (gl_extensions.find("GL_OES_texture_npot") != -1)) {
815271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            mUseNpotTextures = true;
816271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        }
817271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    }
818271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar
819a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    // Blend required to draw time on top of animation frames.
820a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
821a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glShadeModel(GL_FLAT);
822a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_DITHER);
823a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_SCISSOR_TEST);
824a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_BLEND);
825a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
826a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glBindTexture(GL_TEXTURE_2D, 0);
827a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glEnable(GL_TEXTURE_2D);
828a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
829a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
830a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
831a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
832a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
833a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
8340e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    bool clockFontInitialized = false;
835a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    if (mClockEnabled) {
8360e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        clockFontInitialized =
8370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            (initFont(&animation->clockFont, CLOCK_FONT_ASSET) == NO_ERROR);
8380e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        mClockEnabled = clockFontInitialized;
839a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
840a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
8419748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mClockEnabled && !updateIsTimeAccurate()) {
8429748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread = new TimeCheckThread(this);
8439748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread->run("BootAnimation::TimeCheckThread", PRIORITY_NORMAL);
8449748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
84639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    playAnimation(*animation);
8479748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
848a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch    if (mTimeCheckThread != nullptr) {
8499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread->requestExit();
850a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch        mTimeCheckThread = nullptr;
851a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch    }
852a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch
85339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    releaseAnimation(animation);
85439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
8550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (clockFontInitialized) {
8560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glDeleteTextures(1, &animation->clockFont.texture.name);
857815e51d76cc284f3ca59534d9f9372d3dd50f574Andriy Naborskyy    }
858815e51d76cc284f3ca59534d9f9372d3dd50f574Andriy Naborskyy
85939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return false;
86039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
86139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
86239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::playAnimation(const Animation& animation)
86339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
86439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    const size_t pcount = animation.parts.size();
865a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    nsecs_t frameDuration = s2ns(1) / animation.fps;
866dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    const int animationX = (mWidth - animation.width) / 2;
867dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    const int animationY = (mHeight - animation.height) / 2;
8689f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian
869d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park    ALOGD("%sAnimationShownTiming start time: %" PRId64 "ms", mShuttingDown ? "Shutdown" : "Boot",
870d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park            elapsedRealtime());
871afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    for (size_t i=0 ; i<pcount ; i++) {
872a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const Animation::Part& part(animation.parts[i]);
873a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const size_t fcount = part.frames.size();
874a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        glBindTexture(GL_TEXTURE_2D, 0);
875a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
87639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        // Handle animation package
87739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        if (part.animation != NULL) {
87839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            playAnimation(*part.animation);
87939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            if (exitPending())
88039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy                break;
88139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            continue; //to next part
88239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        }
88339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
884a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        for (int r=0 ; !part.count || r<part.count ; r++) {
885d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            // Exit any non playuntil complete parts immediately
886d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            if(exitPending() && !part.playUntilComplete)
887d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                break;
888d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
8897464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne            mCallbacks->playPart(i, part, r);
890ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
891083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            glClearColor(
892083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[0],
893083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[1],
894083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[2],
895083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    1.0f);
896083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
897afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            for (size_t j=0 ; j<fcount && (!exitPending() || part.playUntilComplete) ; j++) {
898a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                const Animation::Frame& frame(part.frames[j]);
899db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                nsecs_t lastFrame = systemTime();
900a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
901a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                if (r > 0) {
902a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    glBindTexture(GL_TEXTURE_2D, frame.tid);
903a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                } else {
904a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    if (part.count != 1) {
905a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glGenTextures(1, &frame.tid);
906a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glBindTexture(GL_TEXTURE_2D, frame.tid);
907a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
908a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
909a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    }
9100e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    int w, h;
9110e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    initTexture(frame.map, &w, &h);
912a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                }
913a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
914dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                const int xc = animationX + frame.trimX;
915dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                const int yc = animationY + frame.trimY;
916dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                Region clearReg(Rect(mWidth, mHeight));
917dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                clearReg.subtractSelf(Rect(xc, yc, xc+frame.trimWidth, yc+frame.trimHeight));
9189f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                if (!clearReg.isEmpty()) {
9199f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    Region::const_iterator head(clearReg.begin());
9209f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    Region::const_iterator tail(clearReg.end());
9219f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    glEnable(GL_SCISSOR_TEST);
9229f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    while (head != tail) {
923cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe                        const Rect& r2(*head++);
924dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                        glScissor(r2.left, mHeight - r2.bottom, r2.width(), r2.height());
9259f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                        glClear(GL_COLOR_BUFFER_BIT);
9269f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    }
9279f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    glDisable(GL_SCISSOR_TEST);
9289f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                }
929dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                // specify the y center as ceiling((mHeight - frame.trimHeight) / 2)
930dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                // which is equivalent to mHeight - (yc + frame.trimHeight)
931dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                glDrawTexiOES(xc, mHeight - (yc + frame.trimHeight),
932dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                              0, frame.trimWidth, frame.trimHeight);
9330e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                if (mClockEnabled && mTimeIsAccurate && validClock(part)) {
9340e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    drawClock(animation.clockFont, part.clockPosX, part.clockPosY);
935a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi                }
936a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
937a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                eglSwapBuffers(mDisplay, mSurface);
938a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
939a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                nsecs_t now = systemTime();
940a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                nsecs_t delay = frameDuration - (now - lastFrame);
941db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                //ALOGD("%lld, %lld", ns2ms(now - lastFrame), ns2ms(delay));
942a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                lastFrame = now;
943db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian
944db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                if (delay > 0) {
945db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    struct timespec spec;
946db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    spec.tv_sec  = (now + delay) / 1000000000;
947db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    spec.tv_nsec = (now + delay) % 1000000000;
948db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    int err;
949db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    do {
950db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                        err = clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &spec, NULL);
951db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    } while (err<0 && errno == EINTR);
952db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                }
953d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
954d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                checkExit();
955a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
956d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
957a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            usleep(part.pause * ns2us(frameDuration));
958d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
959d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            // For infinite parts, we've now played them at least once, so perhaps exit
960d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            if(exitPending() && !part.count)
961d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                break;
962a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
963a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
9642fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    }
9652fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch
9662fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    // Free textures created for looping parts now that the animation is done.
9672fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    for (const Animation::Part& part : animation.parts) {
968a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        if (part.count != 1) {
9692fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch            const size_t fcount = part.frames.size();
9702fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch            for (size_t j = 0; j < fcount; j++) {
971a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                const Animation::Frame& frame(part.frames[j]);
972a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glDeleteTextures(1, &frame.tid);
973a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
974a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
975a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
976d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
97739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
97839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
979a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
98039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyyvoid BootAnimation::releaseAnimation(Animation* animation) const
98139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
98239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    for (Vector<Animation::Part>::iterator it = animation->parts.begin(),
98339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy         e = animation->parts.end(); it != e; ++it) {
98439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        if (it->animation)
98539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            releaseAnimation(it->animation);
98639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
98739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (animation->zip)
98839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        delete animation->zip;
98939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    delete animation;
990a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian}
991a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
99239218ba26d5bb8646f04273f2c3731598721c1daAndriy NaborskyyBootAnimation::Animation* BootAnimation::loadAnimation(const String8& fn)
99339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
99439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (mLoadedFiles.indexOf(fn) >= 0) {
99539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        ALOGE("File \"%s\" is already loaded. Cyclic ref is not allowed",
99639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            fn.string());
99739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return NULL;
99839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
99939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    ZipFileRO *zip = ZipFileRO::open(fn);
100039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (zip == NULL) {
100139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        ALOGE("Failed to open animation zip \"%s\": %s",
100239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            fn.string(), strerror(errno));
100339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return NULL;
1004a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
1005a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
100639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    Animation *animation =  new Animation;
100739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    animation->fileName = fn;
100839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    animation->zip = zip;
10090e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    animation->clockFont.map = nullptr;
101039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    mLoadedFiles.add(animation->fileName);
101139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
101239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    parseAnimationDesc(*animation);
1013a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    if (!preloadZip(*animation)) {
1014a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch        return NULL;
1015a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    }
1016a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch
1017a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
101839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    mLoadedFiles.remove(fn);
101939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return animation;
102039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
10219748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10229748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::updateIsTimeAccurate() {
10239748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr long long MAX_TIME_IN_PAST =   60000LL * 60LL * 24LL * 30LL;  // 30 days
10249748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr long long MAX_TIME_IN_FUTURE = 60000LL * 90LL;  // 90 minutes
10259748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10269748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mTimeIsAccurate) {
10279748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10289748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
1029b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    if (mShuttingDown) return true;
10309748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    struct stat statResult;
10319071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi
10329071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    if(stat(TIME_FORMAT_12_HOUR_FLAG_FILE_PATH, &statResult) == 0) {
10339071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi        mTimeFormat12Hour = true;
10349071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    }
10359071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi
10369748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if(stat(ACCURATE_TIME_FLAG_FILE_PATH, &statResult) == 0) {
10379748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeIsAccurate = true;
10389748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10399748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10409748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10419748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    FILE* file = fopen(LAST_TIME_CHANGED_FILE_PATH, "r");
10429748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (file != NULL) {
10439748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      long long lastChangedTime = 0;
10449748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      fscanf(file, "%lld", &lastChangedTime);
10459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      fclose(file);
10469748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      if (lastChangedTime > 0) {
10479748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        struct timespec now;
10489748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        clock_gettime(CLOCK_REALTIME, &now);
10499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        // Match the Java timestamp format
10509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        long long rtcNow = (now.tv_sec * 1000LL) + (now.tv_nsec / 1000000LL);
10519676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi        if (ACCURATE_TIME_EPOCH < rtcNow
10529676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi            && lastChangedTime > (rtcNow - MAX_TIME_IN_PAST)
10539676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi            && lastChangedTime < (rtcNow + MAX_TIME_IN_FUTURE)) {
10549748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            mTimeIsAccurate = true;
10559748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
10569748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      }
10579748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10589748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10599748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return mTimeIsAccurate;
10609748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10619748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10629748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::TimeCheckThread::TimeCheckThread(BootAnimation* bootAnimation) : Thread(false),
10639748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mInotifyFd(-1), mSystemWd(-1), mTimeWd(-1), mBootAnimation(bootAnimation) {}
10649748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10659748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::TimeCheckThread::~TimeCheckThread() {
10669748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    // mInotifyFd may be -1 but that's ok since we're not at risk of attempting to close a valid FD.
10679748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    close(mInotifyFd);
10689748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10699748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::TimeCheckThread::threadLoop() {
10719748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    bool shouldLoop = doThreadLoop() && !mBootAnimation->mTimeIsAccurate
10729748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        && mBootAnimation->mClockEnabled;
10739748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (!shouldLoop) {
10749748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
10759748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
10769748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10779748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return shouldLoop;
10789748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10799748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10809748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::TimeCheckThread::doThreadLoop() {
10819748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr int BUFF_LEN (10 * (sizeof(struct inotify_event) + NAME_MAX + 1));
10829748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10839748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    // Poll instead of doing a blocking read so the Thread can exit if requested.
10849748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    struct pollfd pfd = { mInotifyFd, POLLIN, 0 };
10859748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    ssize_t pollResult = poll(&pfd, 1, 1000);
10869748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10879748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (pollResult == 0) {
10889748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10899748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    } else if (pollResult < 0) {
10909748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not poll inotify events");
10919748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return false;
10929748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10939748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10949748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    char buff[BUFF_LEN] __attribute__ ((aligned(__alignof__(struct inotify_event))));;
10959748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    ssize_t length = read(mInotifyFd, buff, BUFF_LEN);
10969748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (length == 0) {
10979748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10989748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    } else if (length < 0) {
10999748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not read inotify events");
11009748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return false;
11019748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11029748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11039748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    const struct inotify_event *event;
11049748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    for (char* ptr = buff; ptr < buff + length; ptr += sizeof(struct inotify_event) + event->len) {
11059748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        event = (const struct inotify_event *) ptr;
11069748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        if (event->wd == mSystemWd && strcmp(SYSTEM_TIME_DIR_NAME, event->name) == 0) {
11079748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            addTimeDirWatch();
11089748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        } else if (event->wd == mTimeWd && (strcmp(LAST_TIME_CHANGED_FILE_NAME, event->name) == 0
11099748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi                || strcmp(ACCURATE_TIME_FLAG_FILE_NAME, event->name) == 0)) {
11109748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            return !mBootAnimation->updateIsTimeAccurate();
11119748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
11129748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11139748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11149748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return true;
11159748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11169748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11179748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchivoid BootAnimation::TimeCheckThread::addTimeDirWatch() {
11189748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeWd = inotify_add_watch(mInotifyFd, SYSTEM_TIME_DIR_PATH,
11199748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi                IN_CLOSE_WRITE | IN_MOVED_TO | IN_ATTRIB);
11209748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        if (mTimeWd > 0) {
11219748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            // No need to watch for the time directory to be created if it already exists
11229748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            inotify_rm_watch(mInotifyFd, mSystemWd);
11239748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            mSystemWd = -1;
11249748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
11259748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11269748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11279748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatus_t BootAnimation::TimeCheckThread::readyToRun() {
11289748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mInotifyFd = inotify_init();
11299748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mInotifyFd < 0) {
11309748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not initialize inotify fd");
11319748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return NO_INIT;
11329748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11339748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11349748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mSystemWd = inotify_add_watch(mInotifyFd, SYSTEM_DATA_DIR_PATH, IN_CREATE | IN_ATTRIB);
11359748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mSystemWd < 0) {
11369748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11379748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11389748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not add watch for %s", SYSTEM_DATA_DIR_PATH);
11399748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return NO_INIT;
11409748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11419748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11429748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    addTimeDirWatch();
11439748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11449748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mBootAnimation->updateIsTimeAccurate()) {
11459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11469748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11479748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return ALREADY_EXISTS;
11489748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return NO_ERROR;
11519748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11529748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
11549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project; // namespace android
1157