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
32fb7952f57e07c68cc66a3ec69f86694057f89defSteven Moreland#include <cutils/atomic.h>
33bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks#include <cutils/properties.h>
34bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
35b13b9bdad2baf6ad1ec2e56b6b7598fa20f55fc4Mathias Agopian#include <androidfw/AssetManager.h>
36ac31a3b8b09aba1c5ebc73f0cf65cac2210aa6b7Mathias Agopian#include <binder/IPCThreadState.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";
69c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seokstatic const char PRODUCT_BOOTANIMATION_FILE[] = "/product/media/bootanimation.zip";
709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_BOOTANIMATION_FILE[] = "/system/media/bootanimation.zip";
71c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seokstatic const char PRODUCT_ENCRYPTED_BOOTANIMATION_FILE[] = "/product/media/bootanimation-encrypted.zip";
729748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_ENCRYPTED_BOOTANIMATION_FILE[] = "/system/media/bootanimation-encrypted.zip";
73b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Parkstatic const char OEM_SHUTDOWNANIMATION_FILE[] = "/oem/media/shutdownanimation.zip";
74c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seokstatic const char PRODUCT_SHUTDOWNANIMATION_FILE[] = "/product/media/shutdownanimation.zip";
75b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Parkstatic const char SYSTEM_SHUTDOWNANIMATION_FILE[] = "/system/media/shutdownanimation.zip";
76b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park
779748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_DATA_DIR_PATH[] = "/data/system";
789748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_TIME_DIR_NAME[] = "time";
799748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char SYSTEM_TIME_DIR_PATH[] = "/data/system/time";
800e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const char CLOCK_FONT_ASSET[] = "images/clock_font.png";
810e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const char CLOCK_FONT_ZIP_NAME[] = "clock_font.png";
829748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char LAST_TIME_CHANGED_FILE_NAME[] = "last_time_change";
839748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char LAST_TIME_CHANGED_FILE_PATH[] = "/data/system/time/last_time_change";
849748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char ACCURATE_TIME_FLAG_FILE_NAME[] = "time_is_accurate";
859748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char ACCURATE_TIME_FLAG_FILE_PATH[] = "/data/system/time/time_is_accurate";
869071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchistatic const char TIME_FORMAT_12_HOUR_FLAG_FILE_PATH[] = "/data/system/time/time_format_12_hour";
879676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi// Java timestamp format. Don't show the clock if the date is before 2000-01-01 00:00:00.
889676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchistatic const long long ACCURATE_TIME_EPOCH = 946684800000;
890e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr char FONT_BEGIN_CHAR = ' ';
900e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr char FONT_END_CHAR = '~' + 1;
910e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_CHARS = FONT_END_CHAR - FONT_BEGIN_CHAR + 1;
920e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_COLS = 16;
930e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t FONT_NUM_ROWS = FONT_NUM_CHARS / FONT_NUM_COLS;
940e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const int TEXT_CENTER_VALUE = INT_MAX;
950e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic const int TEXT_MISSING_VALUE = INT_MIN;
969748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatic const char EXIT_PROP_NAME[] = "service.bootanim.exit";
97afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamathstatic const int ANIM_ENTRY_NAME_MAX = 256;
980e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatic constexpr size_t TEXT_POS_LEN_MAX = 16;
99afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1027464ac9bd7fe89061e47617e4b6004b88c91d636Ed CoyneBootAnimation::BootAnimation(sp<Callbacks> callbacks)
1032c9e94aa3e9614f0d202ee008c08a9011fa7f75fEd Coyne        : Thread(false), mClockEnabled(true), mTimeIsAccurate(false),
1047464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne        mTimeFormat12Hour(false), mTimeCheckThread(NULL), mCallbacks(callbacks) {
105627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    mSession = new SurfaceComposerClient();
106290c4350edc66de5663e27c54e694c8ee8cf0e0eGeoffrey Pitsch
107b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    std::string powerCtl = android::base::GetProperty("sys.powerctl", "");
108b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    if (powerCtl.empty()) {
109b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        mShuttingDown = false;
110b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    } else {
111b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        mShuttingDown = true;
112b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    }
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid BootAnimation::onFirstRef() {
116bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    status_t err = mSession->linkToComposerDeath(this);
1173762c311729fe9f3af085c14c5c1fb471d994c03Steve Block    ALOGE_IF(err, "linkToComposerDeath failed (%s) ", strerror(-err));
1188434c5369304e639efe8eab368ca410c589d87c2Mathias Agopian    if (err == NO_ERROR) {
119bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian        run("BootAnimation", PRIORITY_DISPLAY);
120bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    }
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
123bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopiansp<SurfaceComposerClient> BootAnimation::session() const {
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mSession;
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
127bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
128afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamathvoid BootAnimation::binderDied(const wp<IBinder>&)
129bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian{
130bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // woah, surfaceflinger died!
1315baa3a62a97544669fba6d65a11c07f252e654ddSteve Block    ALOGD("SurfaceFlinger died, exiting...");
132bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
133bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // calling requestExit() is not enough here because the Surface code
134bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    // might be blocked on a condition variable that will never be updated.
135bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    kill( getpid(), SIGKILL );
136bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian    requestExit();
137bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian}
138bc7261130a51dc9f3461d3970eee1b923bcbf193Mathias Agopian
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t BootAnimation::initTexture(Texture* texture, AssetManager& assets,
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const char* name) {
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Asset* asset = assets.open(name, Asset::ACCESS_BUFFER);
142cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe    if (asset == NULL)
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_INIT;
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    SkBitmap bitmap;
1458898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkData> data = SkData::MakeWithoutCopy(asset->getBuffer(false),
1468898c16987831db47542a92151ed7209e22268aaMatt Sarett            asset->getLength());
1478898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkImage> image = SkImage::MakeFromEncoded(data);
1488898c16987831db47542a92151ed7209e22268aaMatt Sarett    image->asLegacyBitmap(&bitmap, SkImage::kRO_LegacyBitmapMode);
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    asset->close();
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete asset;
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int w = bitmap.width();
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int h = bitmap.height();
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const void* p = bitmap.getPixels();
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    GLint crop[4] = { 0, h, w, -h };
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    texture->w = w;
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    texture->h = h;
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &texture->name);
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, texture->name);
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
16342a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed    switch (bitmap.colorType()) {
16442a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kAlpha_8_SkColorType:
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, w, h, 0, GL_ALPHA,
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_BYTE, p);
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
16842a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kARGB_4444_SkColorType:
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_SHORT_4_4_4_4, p);
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
17242a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kN32_SkColorType:
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_BYTE, p);
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
17642a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kRGB_565_SkColorType:
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB,
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    GL_UNSIGNED_SHORT_5_6_5, p);
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        default:
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, crop);
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
1890e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1930e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatus_t BootAnimation::initTexture(FileMap* map, int* width, int* height)
194a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
195a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    SkBitmap bitmap;
196f67b317c102db72182046ec2a2cffe2331017709Damien Bargiacchi    sk_sp<SkData> data = SkData::MakeWithoutCopy(map->getDataPtr(),
197f67b317c102db72182046ec2a2cffe2331017709Damien Bargiacchi            map->getDataLength());
1988898c16987831db47542a92151ed7209e22268aaMatt Sarett    sk_sp<SkImage> image = SkImage::MakeFromEncoded(data);
1998898c16987831db47542a92151ed7209e22268aaMatt Sarett    image->asLegacyBitmap(&bitmap, SkImage::kRO_LegacyBitmapMode);
200a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2010c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // FileMap memory is never released until application exit.
2020c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // Release it now as the texture is already loaded and the memory used for
2030c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko    // the packed resource can be released.
2040e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    delete map;
2050c1eeb3c8ab53954218d3b8f4be08ff21be3e985Mykola Kondratenko
206a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const int w = bitmap.width();
207a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const int h = bitmap.height();
208a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const void* p = bitmap.getPixels();
209a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
210a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    GLint crop[4] = { 0, h, w, -h };
211a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    int tw = 1 << (31 - __builtin_clz(w));
212a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    int th = 1 << (31 - __builtin_clz(h));
213a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    if (tw < w) tw <<= 1;
214a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    if (th < h) th <<= 1;
215a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
21642a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed    switch (bitmap.colorType()) {
21742a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kN32_SkColorType:
218271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            if (!mUseNpotTextures && (tw != w || th != h)) {
219a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, tw, th, 0, GL_RGBA,
220a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_BYTE, 0);
221a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexSubImage2D(GL_TEXTURE_2D, 0,
222a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        0, 0, w, h, GL_RGBA, GL_UNSIGNED_BYTE, p);
223a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            } else {
224271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA,
225a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_BYTE, p);
226a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
227a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
228a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
22942a1d08df7d417fd4e67eabc91ff05ee77fd9995Mike Reed        case kRGB_565_SkColorType:
230271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            if (!mUseNpotTextures && (tw != w || th != h)) {
231a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, tw, th, 0, GL_RGB,
232a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_SHORT_5_6_5, 0);
233a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glTexSubImage2D(GL_TEXTURE_2D, 0,
234a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        0, 0, w, h, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, p);
235a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            } else {
236271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar                glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB,
237a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        GL_UNSIGNED_SHORT_5_6_5, p);
238a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
239a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
240a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        default:
241a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            break;
242a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
243a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
244a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, crop);
245a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *width = w;
2470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *height = h;
2480e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
249a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    return NO_ERROR;
250a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian}
251a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t BootAnimation::readyToRun() {
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mAssets.addDefaultAssets();
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2550b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    sp<IBinder> dtoken(SurfaceComposerClient::getBuiltInDisplay(
2560b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown            ISurfaceComposer::eDisplayIdMain));
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    DisplayInfo dinfo;
2580b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    status_t status = SurfaceComposerClient::getDisplayInfo(dtoken, &dinfo);
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (status)
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return -1;
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // create the native surface
2630b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown    sp<SurfaceControl> control = session()->createSurface(String8("BootAnimation"),
2640b722fe9ce98d97dbcb6fefd170b85ab7037e528Jeff Brown            dinfo.w, dinfo.h, PIXEL_FORMAT_RGB_565);
265439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian
266e13b58e15b561b1b85788800e0c3af48fa80463eRobert Carr    SurfaceComposerClient::Transaction t;
267e13b58e15b561b1b85788800e0c3af48fa80463eRobert Carr    t.setLayer(control, 0x40000000)
268e13b58e15b561b1b85788800e0c3af48fa80463eRobert Carr        .apply();
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
27017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    sp<Surface> s = control->getSurface();
27117f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize opengl and egl
273738b9a45399485893fa9541fae19c47e01bc4af4Mathias Agopian    const EGLint attribs[] = {
2741b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_RED_SIZE,   8,
2751b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_GREEN_SIZE, 8,
2761b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian            EGL_BLUE_SIZE,  8,
277a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            EGL_DEPTH_SIZE, 0,
278a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            EGL_NONE
279738b9a45399485893fa9541fae19c47e01bc4af4Mathias Agopian    };
280cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe    EGLint w, h;
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLint numConfigs;
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLConfig config;
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLSurface surface;
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLContext context;
285627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
287627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
288627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    eglInitialize(display, 0, 0);
2891b253b702a356619ae3c5e68ee92062b5305c5d7Mathias Agopian    eglChooseConfig(display, attribs, &config, 1, &numConfigs);
2901473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    surface = eglCreateWindowSurface(display, config, s.get(), NULL);
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    context = eglCreateContext(display, config, NULL, NULL);
2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglQuerySurface(display, surface, EGL_WIDTH, &w);
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglQuerySurface(display, surface, EGL_HEIGHT, &h);
294a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
295abac01052c1c40f8582fe9f010c2efe6013e25c6Mathias Agopian    if (eglMakeCurrent(display, surface, surface, context) == EGL_FALSE)
296abac01052c1c40f8582fe9f010c2efe6013e25c6Mathias Agopian        return NO_INIT;
297a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDisplay = display;
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mContext = context;
3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSurface = surface;
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mWidth = w;
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mHeight = h;
30317f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    mFlingerSurfaceControl = control;
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFlingerSurface = s;
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
306c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes    // If the device has encryption turned on or is in process
307bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    // of being encrypted we show the encrypted boot animation.
308bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    char decrypt[PROPERTY_VALUE_MAX];
309bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    property_get("vold.decrypt", decrypt, "");
310bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
311b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    bool encryptedAnimation = atoi(decrypt) != 0 ||
312b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        !strcmp("trigger_restart_min_framework", decrypt);
313bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks
314c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok    if (!mShuttingDown && encryptedAnimation) {
315c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok        static const char* encryptedBootFiles[] =
316c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok            {PRODUCT_ENCRYPTED_BOOTANIMATION_FILE, SYSTEM_ENCRYPTED_BOOTANIMATION_FILE};
317c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok        for (const char* f : encryptedBootFiles) {
318c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok            if (access(f, R_OK) == 0) {
319c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok                mZipFileName = f;
320c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok                return NO_ERROR;
321c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok            }
322c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok        }
32339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
324c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok    static const char* bootFiles[] =
325c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok        {PRODUCT_BOOTANIMATION_FILE, OEM_BOOTANIMATION_FILE, SYSTEM_BOOTANIMATION_FILE};
326b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    static const char* shutdownFiles[] =
327c521bb33acf9312192785469f56ae9ad718fb602Jaekyun Seok        {PRODUCT_SHUTDOWNANIMATION_FILE, OEM_SHUTDOWNANIMATION_FILE, SYSTEM_SHUTDOWNANIMATION_FILE};
328b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park
329b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    for (const char* f : (!mShuttingDown ? bootFiles : shutdownFiles)) {
330b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        if (access(f, R_OK) == 0) {
331b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park            mZipFileName = f;
332b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park            return NO_ERROR;
333b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park        }
334bd9a08d318f94a3d0b2f8f9d7b16b10736d80c5cJason parks    }
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
338a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopianbool BootAnimation::threadLoop()
339a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
340a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    bool r;
341afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    // We have no bootanimation file, so we use the stock android logo
342afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    // animation.
34339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (mZipFileName.isEmpty()) {
344a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        r = android();
345a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    } else {
346a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        r = movie();
347a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
348a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglMakeCurrent(mDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglDestroyContext(mDisplay, mContext);
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglDestroySurface(mDisplay, mSurface);
3526cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    mFlingerSurface.clear();
35317f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    mFlingerSurfaceControl.clear();
354627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    eglTerminate(mDisplay);
3553eee9fbb6df1dbbcefe1f0dfdf1295c95ddf3279Sai Kiran Korwar    eglReleaseThread();
356627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    IPCThreadState::self()->stopProcess();
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return r;
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
360a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopianbool BootAnimation::android()
361a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
362d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park    ALOGD("%sAnimationShownTiming start time: %" PRId64 "ms", mShuttingDown ? "Shutdown" : "Boot",
363d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park            elapsedRealtime());
364b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    initTexture(&mAndroid[0], mAssets, "images/android-logo-mask.png");
365b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    initTexture(&mAndroid[1], mAssets, "images/android-logo-shine.png");
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3677464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne    mCallbacks->init({});
3687464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear screen
370a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glShadeModel(GL_FLAT);
371b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glDisable(GL_DITHER);
372b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glDisable(GL_SCISSOR_TEST);
37359f19e48c1c043bb9debdc35d166e397e2125d33Mathias Agopian    glClearColor(0,0,0,1);
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glClear(GL_COLOR_BUFFER_BIT);
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    eglSwapBuffers(mDisplay, mSurface);
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
377a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glEnable(GL_TEXTURE_2D);
378a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
379a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
380b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const GLint xc = (mWidth  - mAndroid[0].w) / 2;
381b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const GLint yc = (mHeight - mAndroid[0].h) / 2;
382b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    const Rect updateRect(xc, yc, xc + mAndroid[0].w, yc + mAndroid[0].h);
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glScissor(updateRect.left, mHeight - updateRect.bottom, updateRect.width(),
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            updateRect.height());
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
387b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    // Blend state
388b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
389b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
390b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t startTime = systemTime();
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    do {
3931379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        nsecs_t now = systemTime();
3941379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        double time = now - startTime;
395b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        float t = 4.0f * float(time / us2ns(16667)) / mAndroid[1].w;
396b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        GLint offset = (1 - (t - floorf(t))) * mAndroid[1].w;
397b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        GLint x = xc - offset;
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3998166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glDisable(GL_SCISSOR_TEST);
4008166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
4018166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian
4028166864e42bc20c1544b2028e8dd65b90f29a0eaMathias Agopian        glEnable(GL_SCISSOR_TEST);
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisable(GL_BLEND);
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mAndroid[1].name);
405b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(x,                 yc, 0, mAndroid[1].w, mAndroid[1].h);
406b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(x + mAndroid[1].w, yc, 0, mAndroid[1].w, mAndroid[1].h);
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
408b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glEnable(GL_BLEND);
409b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glBindTexture(GL_TEXTURE_2D, mAndroid[0].name);
410b2cf954cb47a73065aa4b7fed753877615614f41Mathias Agopian        glDrawTexiOES(xc, yc, 0, mAndroid[0].w, mAndroid[0].h);
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
412627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian        EGLBoolean res = eglSwapBuffers(mDisplay, mSurface);
413627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian        if (res == EGL_FALSE)
414627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            break;
415627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
4161379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        // 12fps: don't animate too fast to preserve CPU
4171379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        const nsecs_t sleepTime = 83333 - ns2us(systemTime() - now);
4181379665abdb8330caf813fb77902914d1545bcd9Mathias Agopian        if (sleepTime > 0)
419a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            usleep(sleepTime);
420d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
421d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        checkExit();
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } while (!exitPending());
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mAndroid[0].name);
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mAndroid[1].name);
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return false;
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
429d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hestervoid BootAnimation::checkExit() {
430d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    // Allow surface flinger to gracefully request shutdown
431d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    char value[PROPERTY_VALUE_MAX];
432d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    property_get(EXIT_PROP_NAME, value, "0");
433d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    int exitnow = atoi(value);
434d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    if (exitnow) {
435d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        requestExit();
4367464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne        mCallbacks->shutdown();
437d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester    }
438d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester}
439d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
4400e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchibool BootAnimation::validClock(const Animation::Part& part) {
4410e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return part.clockPosX != TEXT_MISSING_VALUE && part.clockPosY != TEXT_MISSING_VALUE;
4420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchibool parseTextCoord(const char* str, int* dest) {
4450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (strcmp("c", str) == 0) {
4460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *dest = TEXT_CENTER_VALUE;
4470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return true;
4480e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4490e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4500e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    char* end;
4510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int val = (int) strtol(str, &end, 0);
4520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (end == str || *end != '\0' || val == INT_MAX || val == INT_MIN) {
4530e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return false;
4540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    *dest = val;
4560e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return true;
4570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4580e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4590e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// Parse two position coordinates. If only string is non-empty, treat it as the y value.
4600e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid parsePosition(const char* str1, const char* str2, int* x, int* y) {
4610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    bool success = false;
4620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (strlen(str1) == 0) {  // No values were specified
4630e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        // success = false
4640e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (strlen(str2) == 0) {  // we have only one value
4650e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (parseTextCoord(str1, y)) {
4660e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            *x = TEXT_CENTER_VALUE;
4670e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            success = true;
4680e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        }
4690e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else {
4700e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (parseTextCoord(str1, x) && parseTextCoord(str2, y)) {
4710e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            success = true;
4720e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        }
4730e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4740e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
4750e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (!success) {
4760e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = TEXT_MISSING_VALUE;
4770e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = TEXT_MISSING_VALUE;
4780e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
4790e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
4800e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
481083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// Parse a color represented as an HTML-style 'RRGGBB' string: each pair of
482083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// characters in str is a hex number in [0, 255], which are converted to
483083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// floating point values in the range [0.0, 1.0] and placed in the
484083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// corresponding elements of color.
485083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall//
486083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// If the input string isn't valid, parseColor returns false and color is
487083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall// left unchanged.
488083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hallstatic bool parseColor(const char str[7], float color[3]) {
489083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    float tmpColor[3];
490083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    for (int i = 0; i < 3; i++) {
491083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        int val = 0;
492083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        for (int j = 0; j < 2; j++) {
493083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            val *= 16;
494083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            char c = str[2*i + j];
495083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            if      (c >= '0' && c <= '9') val += c - '0';
496083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else if (c >= 'A' && c <= 'F') val += (c - 'A') + 10;
497083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else if (c >= 'a' && c <= 'f') val += (c - 'a') + 10;
498083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            else                           return false;
499083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        }
500083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        tmpColor[i] = static_cast<float>(val) / 255.0f;
501083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    }
502083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    memcpy(color, tmpColor, sizeof(tmpColor));
503083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall    return true;
504083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall}
505083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
50639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
50739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyystatic bool readFile(ZipFileRO* zip, const char* name, String8& outString)
508a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian{
50939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    ZipEntryRO entry = zip->findEntryByName(name);
510ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    ALOGE_IF(!entry, "couldn't find %s", name);
511ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    if (!entry) {
512afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        return false;
513afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    }
514a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
51539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    FileMap* entryMap = zip->createEntryFileMap(entry);
51639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    zip->releaseEntry(entry);
517ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    ALOGE_IF(!entryMap, "entryMap is null");
518ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    if (!entryMap) {
519a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        return false;
520a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
521a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
522ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    outString.setTo((char const*)entryMap->getDataPtr(), entryMap->getDataLength());
523688ff4cf0f93d1a886437884f23874e5595d59a3Narayan Kamath    delete entryMap;
524ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    return true;
525ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood}
526ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
5270e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// The font image should be a 96x2 array of character images.  The
5280e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// columns are the printable ASCII characters 0x20 - 0x7f.  The
5290e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi// top row is regular text; the bottom row is bold.
5300e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchistatus_t BootAnimation::initFont(Font* font, const char* fallback) {
5310e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    status_t status = NO_ERROR;
532a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5330e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (font->map != nullptr) {
5340e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glGenTextures(1, &font->texture.name);
5350e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glBindTexture(GL_TEXTURE_2D, font->texture.name);
536a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5370e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        status = initTexture(font->map, &font->texture.w, &font->texture.h);
5380e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5390e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
5400e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
5410e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
5420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
5430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (fallback != nullptr) {
5440e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        status = initTexture(&font->texture, mAssets, fallback);
5450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else {
5460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return NO_INIT;
547a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
548a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5490e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (status == NO_ERROR) {
5500e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        font->char_width = font->texture.w / FONT_NUM_COLS;
5510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        font->char_height = font->texture.h / FONT_NUM_ROWS / 2;  // There are bold and regular rows
5520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
553a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5540e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    return status;
5550e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
556a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5570e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid BootAnimation::drawText(const char* str, const Font& font, bool bold, int* x, int* y) {
5580e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    glEnable(GL_BLEND);  // Allow us to draw on top of the animation
5590e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    glBindTexture(GL_TEXTURE_2D, font.texture.name);
5600e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    const int len = strlen(str);
5620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    const int strWidth = font.char_width * len;
5630e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
5640e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (*x == TEXT_CENTER_VALUE) {
5650e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = (mWidth - strWidth) / 2;
5660e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (*x < 0) {
5670e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x = mWidth + *x - strWidth;
5680e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
5690e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (*y == TEXT_CENTER_VALUE) {
5700e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = (mHeight - font.char_height) / 2;
5710e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    } else if (*y < 0) {
5720e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *y = mHeight + *y - font.char_height;
573a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
574a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5750e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int cropRect[4] = { 0, 0, font.char_width, -font.char_height };
576a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5770e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    for (int i = 0; i < len; i++) {
5780e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char c = str[i];
579a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5800e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if (c < FONT_BEGIN_CHAR || c > FONT_END_CHAR) {
5810e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            c = '?';
582a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        }
583a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
584a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        // Crop the texture to only the pixels in the current glyph
5850e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int charPos = (c - FONT_BEGIN_CHAR);  // Position in the list of valid characters
5860e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int row = charPos / FONT_NUM_COLS;
5870e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        const int col = charPos % FONT_NUM_COLS;
5880e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[0] = col * font.char_width;  // Left of column
5890e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[1] = row * font.char_height * 2; // Top of row
5900e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        // Move down to bottom of regular (one char_heigh) or bold (two char_heigh) line
5910e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        cropRect[1] += bold ? 2 * font.char_height : font.char_height;
592a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_CROP_RECT_OES, cropRect);
593a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5940e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glDrawTexiOES(*x, *y, 0, font.char_width, font.char_height);
595a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
5960e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        *x += font.char_width;
597a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
598a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
599a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glDisable(GL_BLEND);  // Return to the animation's default behaviour
600a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glBindTexture(GL_TEXTURE_2D, 0);
601a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi}
602a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
6039071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi// We render 12 or 24 hour time.
6040e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchivoid BootAnimation::drawClock(const Font& font, const int xPos, const int yPos) {
6059071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    static constexpr char TIME_FORMAT_12[] = "%l:%M";
6069071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    static constexpr char TIME_FORMAT_24[] = "%H:%M";
6070e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    static constexpr int TIME_LENGTH = 6;
6080e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6090e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    time_t rawtime;
6100e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    time(&rawtime);
6110e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    struct tm* timeInfo = localtime(&rawtime);
6120e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6130e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    char timeBuff[TIME_LENGTH];
6149071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    const char* timeFormat = mTimeFormat12Hour ? TIME_FORMAT_12 : TIME_FORMAT_24;
6159071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    size_t length = strftime(timeBuff, TIME_LENGTH, timeFormat, timeInfo);
6160e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
6170e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (length != TIME_LENGTH - 1) {
6180e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        ALOGE("Couldn't format time; abandoning boot animation clock");
6190e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        mClockEnabled = false;
6200e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        return;
6210e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    }
6220e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
62345a764400485fb6fb9b0d3566e24feac446e616cDamien Bargiacchi    char* out = timeBuff[0] == ' ' ? &timeBuff[1] : &timeBuff[0];
6240e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int x = xPos;
6250e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    int y = yPos;
62645a764400485fb6fb9b0d3566e24feac446e616cDamien Bargiacchi    drawText(out, font, false, &x, &y);
6270e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi}
6280e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
62939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::parseAnimationDesc(Animation& animation)
630ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood{
631ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    String8 desString;
632ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
63339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (!readFile(animation.zip, "desc.txt", desString)) {
634ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood        return false;
635ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood    }
636a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    char const* s = desString.string();
637a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
638a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // Parse the description file
639a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    for (;;) {
640a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const char* endl = strstr(s, "\n");
641cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe        if (endl == NULL) break;
642a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        String8 line(s, endl - s);
643a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const char* l = line.string();
644a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int fps = 0;
645a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int width = 0;
646a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int height = 0;
647a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int count = 0;
648a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi        int pause = 0;
649afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        char path[ANIM_ENTRY_NAME_MAX];
650083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall        char color[7] = "000000"; // default to black if unspecified
6510e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char clockPos1[TEXT_POS_LEN_MAX + 1] = "";
6520e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        char clockPos2[TEXT_POS_LEN_MAX + 1] = "";
653083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
654d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester        char pathType;
655a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        if (sscanf(l, "%d %d %d", &width, &height, &fps) == 3) {
656083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            // ALOGD("> w=%d, h=%d, fps=%d", width, height, fps);
657a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.width = width;
658a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.height = height;
659a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.fps = fps;
6600e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        } else if (sscanf(l, " %c %d %d %s #%6s %16s %16s",
6610e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                          &pathType, &count, &pause, path, color, clockPos1, clockPos2) >= 4) {
6620e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            //ALOGD("> type=%c, count=%d, pause=%d, path=%s, color=%s, clockPos1=%s, clockPos2=%s",
6630e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            //    pathType, count, pause, path, color, clockPos1, clockPos2);
664a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            Animation::Part part;
665d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            part.playUntilComplete = pathType == 'c';
666a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.count = count;
667a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.pause = pause;
668a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            part.path = path;
669d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch            part.audioData = NULL;
67039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.animation = NULL;
671083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            if (!parseColor(color, part.backgroundColor)) {
672083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                ALOGE("> invalid color '#%s'", color);
673083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[0] = 0.0f;
674083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[1] = 0.0f;
675083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                part.backgroundColor[2] = 0.0f;
676083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            }
6770e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            parsePosition(clockPos1, clockPos2, &part.clockPosX, &part.clockPosY);
678a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            animation.parts.add(part);
679a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
68039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        else if (strcmp(l, "$SYSTEM") == 0) {
68139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            // ALOGD("> SYSTEM");
68239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            Animation::Part part;
68339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.playUntilComplete = false;
68439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.count = 1;
68539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.pause = 0;
686d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch            part.audioData = NULL;
68739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            part.animation = loadAnimation(String8(SYSTEM_BOOTANIMATION_FILE));
68839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            if (part.animation != NULL)
68939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy                animation.parts.add(part);
69039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        }
691a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        s = ++endl;
692a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
693a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
69439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
69539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
69639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
69739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::preloadZip(Animation& animation)
69839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
699a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    // read all the data structures
700a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    const size_t pcount = animation.parts.size();
701afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    void *cookie = NULL;
702dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    ZipFileRO* zip = animation.zip;
703dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    if (!zip->startIteration(&cookie)) {
704afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        return false;
705afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    }
706afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
707afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    ZipEntryRO entry;
708afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    char name[ANIM_ENTRY_NAME_MAX];
709dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    while ((entry = zip->nextEntry(cookie)) != NULL) {
710dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        const int foundEntryName = zip->getEntryFileName(entry, name, ANIM_ENTRY_NAME_MAX);
711afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        if (foundEntryName > ANIM_ENTRY_NAME_MAX || foundEntryName == -1) {
712afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            ALOGE("Error fetching entry file name");
713afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            continue;
714afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        }
715afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
716afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 entryName(name);
717afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 path(entryName.getPathDir());
718afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        const String8 leaf(entryName.getPathLeaf());
719afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath        if (leaf.size() > 0) {
7200e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            if (entryName == CLOCK_FONT_ZIP_NAME) {
7210e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                FileMap* map = zip->createEntryFileMap(entry);
7220e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                if (map) {
7230e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    animation.clockFont.map = map;
7240e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                }
7250e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                continue;
7260e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            }
7270e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi
728dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            for (size_t j = 0; j < pcount; j++) {
729afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                if (path == animation.parts[j].path) {
7304600dd053dbdbd4b95f3b11057a1cc55b99f9c77Narayan Kamath                    uint16_t method;
731afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                    // supports only stored png files
732dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                    if (zip->getEntryInfo(entry, &method, NULL, NULL, NULL, NULL, NULL)) {
733afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                        if (method == ZipFileRO::kCompressStored) {
734dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                            FileMap* map = zip->createEntryFileMap(entry);
735afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                            if (map) {
736afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath                                Animation::Part& part(animation.parts.editItemAt(j));
737ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                if (leaf == "audio.wav") {
738ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    // a part may have at most one audio file
739d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                                    part.audioData = (uint8_t *)map->getDataPtr();
740d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                                    part.audioLength = map->getDataLength();
741dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                } else if (leaf == "trim.txt") {
742dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    part.trimData.setTo((char const*)map->getDataPtr(),
743dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                                        map->getDataLength());
744ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                } else {
745ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    Animation::Frame frame;
746ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    frame.name = leaf;
747ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    frame.map = map;
748dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimWidth = animation.width;
749dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimHeight = animation.height;
750dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimX = 0;
751dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                                    frame.trimY = 0;
752ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                    part.frames.add(frame);
753ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood                                }
754a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                            }
755d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                        } else {
756d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch                            ALOGE("bootanimation.zip is compressed; must be only stored");
757a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        }
758a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    }
759a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                }
760a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
761a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
762a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
763a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
764dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    // If there is trimData present, override the positioning defaults.
765dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    for (Animation::Part& part : animation.parts) {
766dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        const char* trimDataStr = part.trimData.string();
767dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        for (size_t frameIdx = 0; frameIdx < part.frames.size(); frameIdx++) {
768dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            const char* endl = strstr(trimDataStr, "\n");
769dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            // No more trimData for this part.
770dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            if (endl == NULL) {
771dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                break;
772dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            }
773dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            String8 line(trimDataStr, endl - trimDataStr);
774dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            const char* lineStr = line.string();
775dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            trimDataStr = ++endl;
776dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            int width = 0, height = 0, x = 0, y = 0;
777dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            if (sscanf(lineStr, "%dx%d+%d+%d", &width, &height, &x, &y) == 4) {
778dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                Animation::Frame& frame(part.frames.editItemAt(frameIdx));
779dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimWidth = width;
780dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimHeight = height;
781dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimX = x;
782dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                frame.trimY = y;
783dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            } else {
784dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                ALOGE("Error parsing trim.txt, line: %s", lineStr);
785dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                break;
786dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch            }
787dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch        }
788dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    }
789dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch
7907464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne    mCallbacks->init(animation.parts);
791d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
792dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    zip->endIteration(cookie);
793afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath
79439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
79539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
79639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
79739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::movie()
79839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
79939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    Animation* animation = loadAnimation(mZipFileName);
80039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (animation == NULL)
80139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return false;
80239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
8039748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    bool anyPartHasClock = false;
8049748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    for (size_t i=0; i < animation->parts.size(); i++) {
8050e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        if(validClock(animation->parts[i])) {
8069748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            anyPartHasClock = true;
8079748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            break;
8089748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
8099748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8109748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (!anyPartHasClock) {
8119748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mClockEnabled = false;
8129748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8139748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
814271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    // Check if npot textures are supported
815271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    mUseNpotTextures = false;
816271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    String8 gl_extensions;
817271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    const char* exts = reinterpret_cast<const char*>(glGetString(GL_EXTENSIONS));
818271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    if (!exts) {
819271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        glGetError();
820271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    } else {
821271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        gl_extensions.setTo(exts);
822271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        if ((gl_extensions.find("GL_ARB_texture_non_power_of_two") != -1) ||
823271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            (gl_extensions.find("GL_OES_texture_npot") != -1)) {
824271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar            mUseNpotTextures = true;
825271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar        }
826271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar    }
827271674930a1f9020a291827a8ecb6ab68782c4faSai Kiran Korwar
828a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    // Blend required to draw time on top of animation frames.
829a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
830a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glShadeModel(GL_FLAT);
831a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_DITHER);
832a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_SCISSOR_TEST);
833a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glDisable(GL_BLEND);
834a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
835a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glBindTexture(GL_TEXTURE_2D, 0);
836a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glEnable(GL_TEXTURE_2D);
837a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
838a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
839a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
840a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
841a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
842a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
8430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    bool clockFontInitialized = false;
844a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    if (mClockEnabled) {
8450e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        clockFontInitialized =
8460e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi            (initFont(&animation->clockFont, CLOCK_FONT_ASSET) == NO_ERROR);
8470e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        mClockEnabled = clockFontInitialized;
848a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
849a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
8509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mClockEnabled && !updateIsTimeAccurate()) {
8519748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread = new TimeCheckThread(this);
8529748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread->run("BootAnimation::TimeCheckThread", PRIORITY_NORMAL);
8539748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
8549748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
85539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    playAnimation(*animation);
8569748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
857a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch    if (mTimeCheckThread != nullptr) {
8589748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeCheckThread->requestExit();
859a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch        mTimeCheckThread = nullptr;
860a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch    }
861a917353da0ed6079d45487a8c96b892c5d754637Geoffrey Pitsch
86239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    releaseAnimation(animation);
86339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
8640e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    if (clockFontInitialized) {
8650e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi        glDeleteTextures(1, &animation->clockFont.texture.name);
866815e51d76cc284f3ca59534d9f9372d3dd50f574Andriy Naborskyy    }
867815e51d76cc284f3ca59534d9f9372d3dd50f574Andriy Naborskyy
86839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return false;
86939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
87039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
87139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyybool BootAnimation::playAnimation(const Animation& animation)
87239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
87339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    const size_t pcount = animation.parts.size();
874a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    nsecs_t frameDuration = s2ns(1) / animation.fps;
875dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    const int animationX = (mWidth - animation.width) / 2;
876dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch    const int animationY = (mHeight - animation.height) / 2;
8779f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian
878d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park    ALOGD("%sAnimationShownTiming start time: %" PRId64 "ms", mShuttingDown ? "Shutdown" : "Boot",
879d1794cdbb4183774ac68e9d445ca87835fbe6461Keun-young Park            elapsedRealtime());
880afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath    for (size_t i=0 ; i<pcount ; i++) {
881a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const Animation::Part& part(animation.parts[i]);
882a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        const size_t fcount = part.frames.size();
883a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        glBindTexture(GL_TEXTURE_2D, 0);
884a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
88539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        // Handle animation package
88639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        if (part.animation != NULL) {
88739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            playAnimation(*part.animation);
88839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            if (exitPending())
88939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy                break;
89039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            continue; //to next part
89139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        }
89239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
893a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        for (int r=0 ; !part.count || r<part.count ; r++) {
894d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            // Exit any non playuntil complete parts immediately
895d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            if(exitPending() && !part.playUntilComplete)
896d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                break;
897d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
8987464ac9bd7fe89061e47617e4b6004b88c91d636Ed Coyne            mCallbacks->playPart(i, part, r);
899ebf9a0d8a888042c16ec0cb6dd8419f18038663fMike Lockwood
900083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall            glClearColor(
901083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[0],
902083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[1],
903083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    part.backgroundColor[2],
904083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall                    1.0f);
905083b84cf8880bb7f0d2c2dc1b45c7d610656dbd1Jesse Hall
906afd31e08299008fdc5c2813f21b2573f29dc53dfNarayan Kamath            for (size_t j=0 ; j<fcount && (!exitPending() || part.playUntilComplete) ; j++) {
907a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                const Animation::Frame& frame(part.frames[j]);
908db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                nsecs_t lastFrame = systemTime();
909a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
910a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                if (r > 0) {
911a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    glBindTexture(GL_TEXTURE_2D, frame.tid);
912a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                } else {
913a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    if (part.count != 1) {
914a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glGenTextures(1, &frame.tid);
915a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glBindTexture(GL_TEXTURE_2D, frame.tid);
916a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
917a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                        glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
918a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                    }
9190e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    int w, h;
9200e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    initTexture(frame.map, &w, &h);
921a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                }
922a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
923dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                const int xc = animationX + frame.trimX;
924dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                const int yc = animationY + frame.trimY;
925dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                Region clearReg(Rect(mWidth, mHeight));
926dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                clearReg.subtractSelf(Rect(xc, yc, xc+frame.trimWidth, yc+frame.trimHeight));
9279f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                if (!clearReg.isEmpty()) {
9289f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    Region::const_iterator head(clearReg.begin());
9299f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    Region::const_iterator tail(clearReg.end());
9309f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    glEnable(GL_SCISSOR_TEST);
9319f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    while (head != tail) {
932cfedceb8c180a2e176154d461659e0c3569dc931Andreas Gampe                        const Rect& r2(*head++);
933dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                        glScissor(r2.left, mHeight - r2.bottom, r2.width(), r2.height());
9349f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                        glClear(GL_COLOR_BUFFER_BIT);
9359f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    }
9369f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                    glDisable(GL_SCISSOR_TEST);
9379f3020db15e50d07ee1ace42e47d90b076f2ce54Mathias Agopian                }
938dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                // specify the y center as ceiling((mHeight - frame.trimHeight) / 2)
939dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                // which is equivalent to mHeight - (yc + frame.trimHeight)
940dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                glDrawTexiOES(xc, mHeight - (yc + frame.trimHeight),
941dd214a796f8b97070645226d6a61f8651f9e13a0Geoffrey Pitsch                              0, frame.trimWidth, frame.trimHeight);
9420e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                if (mClockEnabled && mTimeIsAccurate && validClock(part)) {
9430e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi                    drawClock(animation.clockFont, part.clockPosX, part.clockPosY);
944a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi                }
945a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
946a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                eglSwapBuffers(mDisplay, mSurface);
947a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
948a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                nsecs_t now = systemTime();
949a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                nsecs_t delay = frameDuration - (now - lastFrame);
950db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                //ALOGD("%lld, %lld", ns2ms(now - lastFrame), ns2ms(delay));
951a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                lastFrame = now;
952db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian
953db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                if (delay > 0) {
954db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    struct timespec spec;
955db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    spec.tv_sec  = (now + delay) / 1000000000;
956db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    spec.tv_nsec = (now + delay) % 1000000000;
957db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    int err;
958db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    do {
959db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                        err = clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &spec, NULL);
960db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                    } while (err<0 && errno == EINTR);
961db7dd2af93ae96e5749cfbe978c2695049d9d782Mathias Agopian                }
962d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
963d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                checkExit();
964a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
965d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
966a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            usleep(part.pause * ns2us(frameDuration));
967d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester
968d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            // For infinite parts, we've now played them at least once, so perhaps exit
969d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester            if(exitPending() && !part.count)
970d3782b26b2026e60a8e0d4b967a156369f2a46f8Kevin Hester                break;
971a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
972a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
9732fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    }
9742fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch
9752fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    // Free textures created for looping parts now that the animation is done.
9762fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch    for (const Animation::Part& part : animation.parts) {
977a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        if (part.count != 1) {
9782fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch            const size_t fcount = part.frames.size();
9792fb30fb68e9acb121a3e0dae0cd2790b3a7cc17dGeoffrey Pitsch            for (size_t j = 0; j < fcount; j++) {
980a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                const Animation::Frame& frame(part.frames[j]);
981a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian                glDeleteTextures(1, &frame.tid);
982a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian            }
983a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian        }
984a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian    }
985d6d9a1d0b9cf6fa740d9fe410015b094475c5a4cGeoffrey Pitsch
98639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return true;
98739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
988a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
98939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyyvoid BootAnimation::releaseAnimation(Animation* animation) const
99039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
99139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    for (Vector<Animation::Part>::iterator it = animation->parts.begin(),
99239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy         e = animation->parts.end(); it != e; ++it) {
99339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        if (it->animation)
99439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            releaseAnimation(it->animation);
99539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
99639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (animation->zip)
99739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        delete animation->zip;
99839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    delete animation;
999a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian}
1000a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
100139218ba26d5bb8646f04273f2c3731598721c1daAndriy NaborskyyBootAnimation::Animation* BootAnimation::loadAnimation(const String8& fn)
100239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy{
100339218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (mLoadedFiles.indexOf(fn) >= 0) {
100439218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        ALOGE("File \"%s\" is already loaded. Cyclic ref is not allowed",
100539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            fn.string());
100639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return NULL;
100739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    }
100839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    ZipFileRO *zip = ZipFileRO::open(fn);
100939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    if (zip == NULL) {
101039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        ALOGE("Failed to open animation zip \"%s\": %s",
101139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy            fn.string(), strerror(errno));
101239218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy        return NULL;
1013a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi    }
1014a704b7d3444d218fe3b3f977fd242d2cd626058eDamien Bargiacchi
101539218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    Animation *animation =  new Animation;
101639218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    animation->fileName = fn;
101739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    animation->zip = zip;
10180e3d2ab6d2988a1ae70d13d6d77a0f8109eb66e1Damien Bargiacchi    animation->clockFont.map = nullptr;
101939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    mLoadedFiles.add(animation->fileName);
102039218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy
102139218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    parseAnimationDesc(*animation);
1022a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    if (!preloadZip(*animation)) {
1023a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch        return NULL;
1024a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch    }
1025a91a2d737586ebd0040129333055d8093899751bGeoffrey Pitsch
1026a8826d67097b12cf9d6b14a7e32c0862f17b189aMathias Agopian
102739218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    mLoadedFiles.remove(fn);
102839218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy    return animation;
102939218ba26d5bb8646f04273f2c3731598721c1daAndriy Naborskyy}
10309748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10319748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::updateIsTimeAccurate() {
10329748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr long long MAX_TIME_IN_PAST =   60000LL * 60LL * 24LL * 30LL;  // 30 days
10339748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr long long MAX_TIME_IN_FUTURE = 60000LL * 90LL;  // 90 minutes
10349748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10359748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mTimeIsAccurate) {
10369748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10379748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
1038b593842063770f9e75c623a4bc7bb41b4c1830c6Keun-young Park    if (mShuttingDown) return true;
10399748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    struct stat statResult;
10409071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi
10419071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    if(stat(TIME_FORMAT_12_HOUR_FLAG_FILE_PATH, &statResult) == 0) {
10429071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi        mTimeFormat12Hour = true;
10439071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi    }
10449071db1d370573bad80f127898e34066f56c4e64Damien Bargiacchi
10459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if(stat(ACCURATE_TIME_FLAG_FILE_PATH, &statResult) == 0) {
10469748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeIsAccurate = true;
10479748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10489748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    FILE* file = fopen(LAST_TIME_CHANGED_FILE_PATH, "r");
10519748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (file != NULL) {
10529748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      long long lastChangedTime = 0;
10539748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      fscanf(file, "%lld", &lastChangedTime);
10549748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      fclose(file);
10559748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      if (lastChangedTime > 0) {
10569748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        struct timespec now;
10579748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        clock_gettime(CLOCK_REALTIME, &now);
10589748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        // Match the Java timestamp format
10599748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        long long rtcNow = (now.tv_sec * 1000LL) + (now.tv_nsec / 1000000LL);
10609676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi        if (ACCURATE_TIME_EPOCH < rtcNow
10619676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi            && lastChangedTime > (rtcNow - MAX_TIME_IN_PAST)
10629676281c4f0b34707e34cf9d2f5a866f1979c405Damien Bargiacchi            && lastChangedTime < (rtcNow + MAX_TIME_IN_FUTURE)) {
10639748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            mTimeIsAccurate = true;
10649748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
10659748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi      }
10669748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10679748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10689748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return mTimeIsAccurate;
10699748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10709748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10719748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::TimeCheckThread::TimeCheckThread(BootAnimation* bootAnimation) : Thread(false),
10729748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mInotifyFd(-1), mSystemWd(-1), mTimeWd(-1), mBootAnimation(bootAnimation) {}
10739748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10749748086fe267e21d243a49df1775905094a28dd9Damien BargiacchiBootAnimation::TimeCheckThread::~TimeCheckThread() {
10759748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    // mInotifyFd may be -1 but that's ok since we're not at risk of attempting to close a valid FD.
10769748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    close(mInotifyFd);
10779748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10789748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10799748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::TimeCheckThread::threadLoop() {
10809748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    bool shouldLoop = doThreadLoop() && !mBootAnimation->mTimeIsAccurate
10819748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        && mBootAnimation->mClockEnabled;
10829748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (!shouldLoop) {
10839748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
10849748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
10859748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
10869748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return shouldLoop;
10879748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
10889748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10899748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchibool BootAnimation::TimeCheckThread::doThreadLoop() {
10909748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    static constexpr int BUFF_LEN (10 * (sizeof(struct inotify_event) + NAME_MAX + 1));
10919748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10929748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    // Poll instead of doing a blocking read so the Thread can exit if requested.
10939748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    struct pollfd pfd = { mInotifyFd, POLLIN, 0 };
10949748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    ssize_t pollResult = poll(&pfd, 1, 1000);
10959748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
10969748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (pollResult == 0) {
10979748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
10989748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    } else if (pollResult < 0) {
10999748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not poll inotify events");
11009748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return false;
11019748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11029748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11039748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    char buff[BUFF_LEN] __attribute__ ((aligned(__alignof__(struct inotify_event))));;
11049748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    ssize_t length = read(mInotifyFd, buff, BUFF_LEN);
11059748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (length == 0) {
11069748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return true;
11079748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    } else if (length < 0) {
11089748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not read inotify events");
11099748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return false;
11109748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11119748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11129748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    const struct inotify_event *event;
11139748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    for (char* ptr = buff; ptr < buff + length; ptr += sizeof(struct inotify_event) + event->len) {
11149748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        event = (const struct inotify_event *) ptr;
11159748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        if (event->wd == mSystemWd && strcmp(SYSTEM_TIME_DIR_NAME, event->name) == 0) {
11169748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            addTimeDirWatch();
11179748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        } else if (event->wd == mTimeWd && (strcmp(LAST_TIME_CHANGED_FILE_NAME, event->name) == 0
11189748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi                || strcmp(ACCURATE_TIME_FLAG_FILE_NAME, event->name) == 0)) {
11199748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            return !mBootAnimation->updateIsTimeAccurate();
11209748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
11219748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11229748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11239748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return true;
11249748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11259748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11269748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchivoid BootAnimation::TimeCheckThread::addTimeDirWatch() {
11279748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mTimeWd = inotify_add_watch(mInotifyFd, SYSTEM_TIME_DIR_PATH,
11289748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi                IN_CLOSE_WRITE | IN_MOVED_TO | IN_ATTRIB);
11299748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        if (mTimeWd > 0) {
11309748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            // No need to watch for the time directory to be created if it already exists
11319748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            inotify_rm_watch(mInotifyFd, mSystemWd);
11329748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi            mSystemWd = -1;
11339748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        }
11349748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11359748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11369748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchistatus_t BootAnimation::TimeCheckThread::readyToRun() {
11379748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mInotifyFd = inotify_init();
11389748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mInotifyFd < 0) {
11399748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not initialize inotify fd");
11409748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return NO_INIT;
11419748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11429748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11439748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    mSystemWd = inotify_add_watch(mInotifyFd, SYSTEM_DATA_DIR_PATH, IN_CREATE | IN_ATTRIB);
11449748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mSystemWd < 0) {
11459748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11469748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11479748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        ALOGE("Could not add watch for %s", SYSTEM_DATA_DIR_PATH);
11489748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return NO_INIT;
11499748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11509748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11519748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    addTimeDirWatch();
11529748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11539748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    if (mBootAnimation->updateIsTimeAccurate()) {
11549748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        close(mInotifyFd);
11559748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        mInotifyFd = -1;
11569748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi        return ALREADY_EXISTS;
11579748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    }
11589748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11599748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi    return NO_ERROR;
11609748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi}
11619748086fe267e21d243a49df1775905094a28dd9Damien Bargiacchi
11629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
11639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project; // namespace android
1166