1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker * Copyright (C) 2010 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker *
4*d57664e9SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker *
8*d57664e9SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker *
10*d57664e9SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker */
16*d57664e9SAndroid Build Coastguard Worker
17*d57664e9SAndroid Build Coastguard Worker #include <errno.h>
18*d57664e9SAndroid Build Coastguard Worker #include <fcntl.h>
19*d57664e9SAndroid Build Coastguard Worker #include <stdint.h>
20*d57664e9SAndroid Build Coastguard Worker #include <stdlib.h>
21*d57664e9SAndroid Build Coastguard Worker #include <string.h>
22*d57664e9SAndroid Build Coastguard Worker #include <unistd.h>
23*d57664e9SAndroid Build Coastguard Worker
24*d57664e9SAndroid Build Coastguard Worker #define LOG_TAG "ObbFile"
25*d57664e9SAndroid Build Coastguard Worker
26*d57664e9SAndroid Build Coastguard Worker #include <androidfw/ObbFile.h>
27*d57664e9SAndroid Build Coastguard Worker #include <utils/Compat.h>
28*d57664e9SAndroid Build Coastguard Worker #include <utils/Log.h>
29*d57664e9SAndroid Build Coastguard Worker
30*d57664e9SAndroid Build Coastguard Worker //#define DEBUG 1
31*d57664e9SAndroid Build Coastguard Worker
32*d57664e9SAndroid Build Coastguard Worker #define kFooterTagSize 8 /* last two 32-bit integers */
33*d57664e9SAndroid Build Coastguard Worker
34*d57664e9SAndroid Build Coastguard Worker #define kFooterMinSize 33 /* 32-bit signature version (4 bytes)
35*d57664e9SAndroid Build Coastguard Worker * 32-bit package version (4 bytes)
36*d57664e9SAndroid Build Coastguard Worker * 32-bit flags (4 bytes)
37*d57664e9SAndroid Build Coastguard Worker * 64-bit salt (8 bytes)
38*d57664e9SAndroid Build Coastguard Worker * 32-bit package name size (4 bytes)
39*d57664e9SAndroid Build Coastguard Worker * >=1-character package name (1 byte)
40*d57664e9SAndroid Build Coastguard Worker * 32-bit footer size (4 bytes)
41*d57664e9SAndroid Build Coastguard Worker * 32-bit footer marker (4 bytes)
42*d57664e9SAndroid Build Coastguard Worker */
43*d57664e9SAndroid Build Coastguard Worker
44*d57664e9SAndroid Build Coastguard Worker #define kMaxBufSize 32768 /* Maximum file read buffer */
45*d57664e9SAndroid Build Coastguard Worker
46*d57664e9SAndroid Build Coastguard Worker #define kSignature 0x01059983U /* ObbFile signature */
47*d57664e9SAndroid Build Coastguard Worker
48*d57664e9SAndroid Build Coastguard Worker #define kSigVersion 1 /* We only know about signature version 1 */
49*d57664e9SAndroid Build Coastguard Worker
50*d57664e9SAndroid Build Coastguard Worker /* offsets in version 1 of the header */
51*d57664e9SAndroid Build Coastguard Worker #define kPackageVersionOffset 4
52*d57664e9SAndroid Build Coastguard Worker #define kFlagsOffset 8
53*d57664e9SAndroid Build Coastguard Worker #define kSaltOffset 12
54*d57664e9SAndroid Build Coastguard Worker #define kPackageNameLenOffset 20
55*d57664e9SAndroid Build Coastguard Worker #define kPackageNameOffset 24
56*d57664e9SAndroid Build Coastguard Worker
57*d57664e9SAndroid Build Coastguard Worker /*
58*d57664e9SAndroid Build Coastguard Worker * TEMP_FAILURE_RETRY is defined by some, but not all, versions of
59*d57664e9SAndroid Build Coastguard Worker * <unistd.h>. (Alas, it is not as standard as we'd hoped!) So, if it's
60*d57664e9SAndroid Build Coastguard Worker * not already defined, then define it here.
61*d57664e9SAndroid Build Coastguard Worker */
62*d57664e9SAndroid Build Coastguard Worker #ifndef TEMP_FAILURE_RETRY
63*d57664e9SAndroid Build Coastguard Worker /* Used to retry syscalls that can return EINTR. */
64*d57664e9SAndroid Build Coastguard Worker #define TEMP_FAILURE_RETRY(exp) ({ \
65*d57664e9SAndroid Build Coastguard Worker typeof (exp) _rc; \
66*d57664e9SAndroid Build Coastguard Worker do { \
67*d57664e9SAndroid Build Coastguard Worker _rc = (exp); \
68*d57664e9SAndroid Build Coastguard Worker } while (_rc == -1 && errno == EINTR); \
69*d57664e9SAndroid Build Coastguard Worker _rc; })
70*d57664e9SAndroid Build Coastguard Worker #endif
71*d57664e9SAndroid Build Coastguard Worker
72*d57664e9SAndroid Build Coastguard Worker
73*d57664e9SAndroid Build Coastguard Worker namespace android {
74*d57664e9SAndroid Build Coastguard Worker
ObbFile()75*d57664e9SAndroid Build Coastguard Worker ObbFile::ObbFile()
76*d57664e9SAndroid Build Coastguard Worker : mPackageName("")
77*d57664e9SAndroid Build Coastguard Worker , mVersion(-1)
78*d57664e9SAndroid Build Coastguard Worker , mFlags(0)
79*d57664e9SAndroid Build Coastguard Worker {
80*d57664e9SAndroid Build Coastguard Worker memset(mSalt, 0, sizeof(mSalt));
81*d57664e9SAndroid Build Coastguard Worker }
82*d57664e9SAndroid Build Coastguard Worker
~ObbFile()83*d57664e9SAndroid Build Coastguard Worker ObbFile::~ObbFile() {
84*d57664e9SAndroid Build Coastguard Worker }
85*d57664e9SAndroid Build Coastguard Worker
readFrom(const char * filename)86*d57664e9SAndroid Build Coastguard Worker bool ObbFile::readFrom(const char* filename)
87*d57664e9SAndroid Build Coastguard Worker {
88*d57664e9SAndroid Build Coastguard Worker int fd;
89*d57664e9SAndroid Build Coastguard Worker bool success = false;
90*d57664e9SAndroid Build Coastguard Worker
91*d57664e9SAndroid Build Coastguard Worker fd = ::open(filename, O_RDONLY);
92*d57664e9SAndroid Build Coastguard Worker if (fd < 0) {
93*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't open file %s: %s", filename, strerror(errno));
94*d57664e9SAndroid Build Coastguard Worker goto out;
95*d57664e9SAndroid Build Coastguard Worker }
96*d57664e9SAndroid Build Coastguard Worker success = readFrom(fd);
97*d57664e9SAndroid Build Coastguard Worker close(fd);
98*d57664e9SAndroid Build Coastguard Worker
99*d57664e9SAndroid Build Coastguard Worker if (!success) {
100*d57664e9SAndroid Build Coastguard Worker ALOGW("failed to read from %s (fd=%d)\n", filename, fd);
101*d57664e9SAndroid Build Coastguard Worker }
102*d57664e9SAndroid Build Coastguard Worker
103*d57664e9SAndroid Build Coastguard Worker out:
104*d57664e9SAndroid Build Coastguard Worker return success;
105*d57664e9SAndroid Build Coastguard Worker }
106*d57664e9SAndroid Build Coastguard Worker
readFrom(int fd)107*d57664e9SAndroid Build Coastguard Worker bool ObbFile::readFrom(int fd)
108*d57664e9SAndroid Build Coastguard Worker {
109*d57664e9SAndroid Build Coastguard Worker if (fd < 0) {
110*d57664e9SAndroid Build Coastguard Worker ALOGW("attempt to read from invalid fd\n");
111*d57664e9SAndroid Build Coastguard Worker return false;
112*d57664e9SAndroid Build Coastguard Worker }
113*d57664e9SAndroid Build Coastguard Worker
114*d57664e9SAndroid Build Coastguard Worker return parseObbFile(fd);
115*d57664e9SAndroid Build Coastguard Worker }
116*d57664e9SAndroid Build Coastguard Worker
parseObbFile(int fd)117*d57664e9SAndroid Build Coastguard Worker bool ObbFile::parseObbFile(int fd)
118*d57664e9SAndroid Build Coastguard Worker {
119*d57664e9SAndroid Build Coastguard Worker off64_t fileLength = lseek64(fd, 0, SEEK_END);
120*d57664e9SAndroid Build Coastguard Worker
121*d57664e9SAndroid Build Coastguard Worker if (fileLength < kFooterMinSize) {
122*d57664e9SAndroid Build Coastguard Worker if (fileLength < 0) {
123*d57664e9SAndroid Build Coastguard Worker ALOGW("error seeking in ObbFile: %s\n", strerror(errno));
124*d57664e9SAndroid Build Coastguard Worker } else {
125*d57664e9SAndroid Build Coastguard Worker ALOGW("file is only %lld (less than %d minimum)\n", (long long int)fileLength, kFooterMinSize);
126*d57664e9SAndroid Build Coastguard Worker }
127*d57664e9SAndroid Build Coastguard Worker return false;
128*d57664e9SAndroid Build Coastguard Worker }
129*d57664e9SAndroid Build Coastguard Worker
130*d57664e9SAndroid Build Coastguard Worker ssize_t actual;
131*d57664e9SAndroid Build Coastguard Worker size_t footerSize;
132*d57664e9SAndroid Build Coastguard Worker
133*d57664e9SAndroid Build Coastguard Worker {
134*d57664e9SAndroid Build Coastguard Worker lseek64(fd, fileLength - kFooterTagSize, SEEK_SET);
135*d57664e9SAndroid Build Coastguard Worker
136*d57664e9SAndroid Build Coastguard Worker char footer[kFooterTagSize];
137*d57664e9SAndroid Build Coastguard Worker actual = TEMP_FAILURE_RETRY(read(fd, footer, kFooterTagSize));
138*d57664e9SAndroid Build Coastguard Worker if (actual != kFooterTagSize) {
139*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't read footer signature: %s\n", strerror(errno));
140*d57664e9SAndroid Build Coastguard Worker return false;
141*d57664e9SAndroid Build Coastguard Worker }
142*d57664e9SAndroid Build Coastguard Worker
143*d57664e9SAndroid Build Coastguard Worker unsigned int fileSig = get4LE((unsigned char*)footer + sizeof(int32_t));
144*d57664e9SAndroid Build Coastguard Worker if (fileSig != kSignature) {
145*d57664e9SAndroid Build Coastguard Worker ALOGW("footer didn't match magic string (expected 0x%08x; got 0x%08x)\n",
146*d57664e9SAndroid Build Coastguard Worker kSignature, fileSig);
147*d57664e9SAndroid Build Coastguard Worker return false;
148*d57664e9SAndroid Build Coastguard Worker }
149*d57664e9SAndroid Build Coastguard Worker
150*d57664e9SAndroid Build Coastguard Worker footerSize = get4LE((unsigned char*)footer);
151*d57664e9SAndroid Build Coastguard Worker if (footerSize > (size_t)fileLength - kFooterTagSize
152*d57664e9SAndroid Build Coastguard Worker || footerSize > kMaxBufSize) {
153*d57664e9SAndroid Build Coastguard Worker ALOGW("claimed footer size is too large (0x%08zx; file size is 0x%08lld)\n",
154*d57664e9SAndroid Build Coastguard Worker footerSize, (long long int)fileLength);
155*d57664e9SAndroid Build Coastguard Worker return false;
156*d57664e9SAndroid Build Coastguard Worker }
157*d57664e9SAndroid Build Coastguard Worker
158*d57664e9SAndroid Build Coastguard Worker if (footerSize < (kFooterMinSize - kFooterTagSize)) {
159*d57664e9SAndroid Build Coastguard Worker ALOGW("claimed footer size is too small (0x%zx; minimum size is 0x%x)\n",
160*d57664e9SAndroid Build Coastguard Worker footerSize, kFooterMinSize - kFooterTagSize);
161*d57664e9SAndroid Build Coastguard Worker return false;
162*d57664e9SAndroid Build Coastguard Worker }
163*d57664e9SAndroid Build Coastguard Worker }
164*d57664e9SAndroid Build Coastguard Worker
165*d57664e9SAndroid Build Coastguard Worker off64_t fileOffset = fileLength - footerSize - kFooterTagSize;
166*d57664e9SAndroid Build Coastguard Worker if (lseek64(fd, fileOffset, SEEK_SET) != fileOffset) {
167*d57664e9SAndroid Build Coastguard Worker ALOGW("seek %lld failed: %s\n", (long long int)fileOffset, strerror(errno));
168*d57664e9SAndroid Build Coastguard Worker return false;
169*d57664e9SAndroid Build Coastguard Worker }
170*d57664e9SAndroid Build Coastguard Worker
171*d57664e9SAndroid Build Coastguard Worker mFooterStart = fileOffset;
172*d57664e9SAndroid Build Coastguard Worker
173*d57664e9SAndroid Build Coastguard Worker char* scanBuf = (char*)malloc(footerSize);
174*d57664e9SAndroid Build Coastguard Worker if (scanBuf == NULL) {
175*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't allocate scanBuf: %s\n", strerror(errno));
176*d57664e9SAndroid Build Coastguard Worker return false;
177*d57664e9SAndroid Build Coastguard Worker }
178*d57664e9SAndroid Build Coastguard Worker
179*d57664e9SAndroid Build Coastguard Worker actual = TEMP_FAILURE_RETRY(read(fd, scanBuf, footerSize));
180*d57664e9SAndroid Build Coastguard Worker // readAmount is guaranteed to be less than kMaxBufSize
181*d57664e9SAndroid Build Coastguard Worker if (actual != (ssize_t)footerSize) {
182*d57664e9SAndroid Build Coastguard Worker ALOGI("couldn't read ObbFile footer: %s\n", strerror(errno));
183*d57664e9SAndroid Build Coastguard Worker free(scanBuf);
184*d57664e9SAndroid Build Coastguard Worker return false;
185*d57664e9SAndroid Build Coastguard Worker }
186*d57664e9SAndroid Build Coastguard Worker
187*d57664e9SAndroid Build Coastguard Worker #ifdef DEBUG
188*d57664e9SAndroid Build Coastguard Worker for (int i = 0; i < footerSize; ++i) {
189*d57664e9SAndroid Build Coastguard Worker ALOGI("char: 0x%02x\n", scanBuf[i]);
190*d57664e9SAndroid Build Coastguard Worker }
191*d57664e9SAndroid Build Coastguard Worker #endif
192*d57664e9SAndroid Build Coastguard Worker
193*d57664e9SAndroid Build Coastguard Worker uint32_t sigVersion = get4LE((unsigned char*)scanBuf);
194*d57664e9SAndroid Build Coastguard Worker if (sigVersion != kSigVersion) {
195*d57664e9SAndroid Build Coastguard Worker ALOGW("Unsupported ObbFile version %d\n", sigVersion);
196*d57664e9SAndroid Build Coastguard Worker free(scanBuf);
197*d57664e9SAndroid Build Coastguard Worker return false;
198*d57664e9SAndroid Build Coastguard Worker }
199*d57664e9SAndroid Build Coastguard Worker
200*d57664e9SAndroid Build Coastguard Worker mVersion = (int32_t) get4LE((unsigned char*)scanBuf + kPackageVersionOffset);
201*d57664e9SAndroid Build Coastguard Worker mFlags = (int32_t) get4LE((unsigned char*)scanBuf + kFlagsOffset);
202*d57664e9SAndroid Build Coastguard Worker
203*d57664e9SAndroid Build Coastguard Worker memcpy(&mSalt, (unsigned char*)scanBuf + kSaltOffset, sizeof(mSalt));
204*d57664e9SAndroid Build Coastguard Worker
205*d57664e9SAndroid Build Coastguard Worker size_t packageNameLen = get4LE((unsigned char*)scanBuf + kPackageNameLenOffset);
206*d57664e9SAndroid Build Coastguard Worker if (packageNameLen == 0
207*d57664e9SAndroid Build Coastguard Worker || packageNameLen > (footerSize - kPackageNameOffset)) {
208*d57664e9SAndroid Build Coastguard Worker ALOGW("bad ObbFile package name length (0x%04zx; 0x%04zx possible)\n",
209*d57664e9SAndroid Build Coastguard Worker packageNameLen, footerSize - kPackageNameOffset);
210*d57664e9SAndroid Build Coastguard Worker free(scanBuf);
211*d57664e9SAndroid Build Coastguard Worker return false;
212*d57664e9SAndroid Build Coastguard Worker }
213*d57664e9SAndroid Build Coastguard Worker
214*d57664e9SAndroid Build Coastguard Worker char* packageName = reinterpret_cast<char*>(scanBuf + kPackageNameOffset);
215*d57664e9SAndroid Build Coastguard Worker mPackageName = String8(const_cast<char*>(packageName), packageNameLen);
216*d57664e9SAndroid Build Coastguard Worker
217*d57664e9SAndroid Build Coastguard Worker free(scanBuf);
218*d57664e9SAndroid Build Coastguard Worker
219*d57664e9SAndroid Build Coastguard Worker #ifdef DEBUG
220*d57664e9SAndroid Build Coastguard Worker ALOGI("Obb scan succeeded: packageName=%s, version=%d\n", mPackageName.c_str(), mVersion);
221*d57664e9SAndroid Build Coastguard Worker #endif
222*d57664e9SAndroid Build Coastguard Worker
223*d57664e9SAndroid Build Coastguard Worker return true;
224*d57664e9SAndroid Build Coastguard Worker }
225*d57664e9SAndroid Build Coastguard Worker
writeTo(const char * filename)226*d57664e9SAndroid Build Coastguard Worker bool ObbFile::writeTo(const char* filename)
227*d57664e9SAndroid Build Coastguard Worker {
228*d57664e9SAndroid Build Coastguard Worker int fd;
229*d57664e9SAndroid Build Coastguard Worker bool success = false;
230*d57664e9SAndroid Build Coastguard Worker
231*d57664e9SAndroid Build Coastguard Worker fd = ::open(filename, O_WRONLY);
232*d57664e9SAndroid Build Coastguard Worker if (fd < 0) {
233*d57664e9SAndroid Build Coastguard Worker goto out;
234*d57664e9SAndroid Build Coastguard Worker }
235*d57664e9SAndroid Build Coastguard Worker success = writeTo(fd);
236*d57664e9SAndroid Build Coastguard Worker close(fd);
237*d57664e9SAndroid Build Coastguard Worker
238*d57664e9SAndroid Build Coastguard Worker out:
239*d57664e9SAndroid Build Coastguard Worker if (!success) {
240*d57664e9SAndroid Build Coastguard Worker ALOGW("failed to write to %s: %s\n", filename, strerror(errno));
241*d57664e9SAndroid Build Coastguard Worker }
242*d57664e9SAndroid Build Coastguard Worker return success;
243*d57664e9SAndroid Build Coastguard Worker }
244*d57664e9SAndroid Build Coastguard Worker
writeTo(int fd)245*d57664e9SAndroid Build Coastguard Worker bool ObbFile::writeTo(int fd)
246*d57664e9SAndroid Build Coastguard Worker {
247*d57664e9SAndroid Build Coastguard Worker if (fd < 0) {
248*d57664e9SAndroid Build Coastguard Worker return false;
249*d57664e9SAndroid Build Coastguard Worker }
250*d57664e9SAndroid Build Coastguard Worker
251*d57664e9SAndroid Build Coastguard Worker lseek64(fd, 0, SEEK_END);
252*d57664e9SAndroid Build Coastguard Worker
253*d57664e9SAndroid Build Coastguard Worker if (mPackageName.size() == 0 || mVersion == -1) {
254*d57664e9SAndroid Build Coastguard Worker ALOGW("tried to write uninitialized ObbFile data\n");
255*d57664e9SAndroid Build Coastguard Worker return false;
256*d57664e9SAndroid Build Coastguard Worker }
257*d57664e9SAndroid Build Coastguard Worker
258*d57664e9SAndroid Build Coastguard Worker unsigned char intBuf[sizeof(uint32_t)+1];
259*d57664e9SAndroid Build Coastguard Worker memset(&intBuf, 0, sizeof(intBuf));
260*d57664e9SAndroid Build Coastguard Worker
261*d57664e9SAndroid Build Coastguard Worker put4LE(intBuf, kSigVersion);
262*d57664e9SAndroid Build Coastguard Worker if (write(fd, &intBuf, sizeof(uint32_t)) != (ssize_t)sizeof(uint32_t)) {
263*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write signature version: %s\n", strerror(errno));
264*d57664e9SAndroid Build Coastguard Worker return false;
265*d57664e9SAndroid Build Coastguard Worker }
266*d57664e9SAndroid Build Coastguard Worker
267*d57664e9SAndroid Build Coastguard Worker put4LE(intBuf, mVersion);
268*d57664e9SAndroid Build Coastguard Worker if (write(fd, &intBuf, sizeof(uint32_t)) != (ssize_t)sizeof(uint32_t)) {
269*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write package version\n");
270*d57664e9SAndroid Build Coastguard Worker return false;
271*d57664e9SAndroid Build Coastguard Worker }
272*d57664e9SAndroid Build Coastguard Worker
273*d57664e9SAndroid Build Coastguard Worker put4LE(intBuf, mFlags);
274*d57664e9SAndroid Build Coastguard Worker if (write(fd, &intBuf, sizeof(uint32_t)) != (ssize_t)sizeof(uint32_t)) {
275*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write package version\n");
276*d57664e9SAndroid Build Coastguard Worker return false;
277*d57664e9SAndroid Build Coastguard Worker }
278*d57664e9SAndroid Build Coastguard Worker
279*d57664e9SAndroid Build Coastguard Worker if (write(fd, mSalt, sizeof(mSalt)) != (ssize_t)sizeof(mSalt)) {
280*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write salt: %s\n", strerror(errno));
281*d57664e9SAndroid Build Coastguard Worker return false;
282*d57664e9SAndroid Build Coastguard Worker }
283*d57664e9SAndroid Build Coastguard Worker
284*d57664e9SAndroid Build Coastguard Worker size_t packageNameLen = mPackageName.size();
285*d57664e9SAndroid Build Coastguard Worker put4LE(intBuf, packageNameLen);
286*d57664e9SAndroid Build Coastguard Worker if (write(fd, &intBuf, sizeof(uint32_t)) != (ssize_t)sizeof(uint32_t)) {
287*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write package name length: %s\n", strerror(errno));
288*d57664e9SAndroid Build Coastguard Worker return false;
289*d57664e9SAndroid Build Coastguard Worker }
290*d57664e9SAndroid Build Coastguard Worker
291*d57664e9SAndroid Build Coastguard Worker if (write(fd, mPackageName.c_str(), packageNameLen) != (ssize_t)packageNameLen) {
292*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write package name: %s\n", strerror(errno));
293*d57664e9SAndroid Build Coastguard Worker return false;
294*d57664e9SAndroid Build Coastguard Worker }
295*d57664e9SAndroid Build Coastguard Worker
296*d57664e9SAndroid Build Coastguard Worker put4LE(intBuf, kPackageNameOffset + packageNameLen);
297*d57664e9SAndroid Build Coastguard Worker if (write(fd, &intBuf, sizeof(uint32_t)) != (ssize_t)sizeof(uint32_t)) {
298*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write footer size: %s\n", strerror(errno));
299*d57664e9SAndroid Build Coastguard Worker return false;
300*d57664e9SAndroid Build Coastguard Worker }
301*d57664e9SAndroid Build Coastguard Worker
302*d57664e9SAndroid Build Coastguard Worker put4LE(intBuf, kSignature);
303*d57664e9SAndroid Build Coastguard Worker if (write(fd, &intBuf, sizeof(uint32_t)) != (ssize_t)sizeof(uint32_t)) {
304*d57664e9SAndroid Build Coastguard Worker ALOGW("couldn't write footer magic signature: %s\n", strerror(errno));
305*d57664e9SAndroid Build Coastguard Worker return false;
306*d57664e9SAndroid Build Coastguard Worker }
307*d57664e9SAndroid Build Coastguard Worker
308*d57664e9SAndroid Build Coastguard Worker return true;
309*d57664e9SAndroid Build Coastguard Worker }
310*d57664e9SAndroid Build Coastguard Worker
removeFrom(const char * filename)311*d57664e9SAndroid Build Coastguard Worker bool ObbFile::removeFrom(const char* filename)
312*d57664e9SAndroid Build Coastguard Worker {
313*d57664e9SAndroid Build Coastguard Worker int fd;
314*d57664e9SAndroid Build Coastguard Worker bool success = false;
315*d57664e9SAndroid Build Coastguard Worker
316*d57664e9SAndroid Build Coastguard Worker fd = ::open(filename, O_RDWR);
317*d57664e9SAndroid Build Coastguard Worker if (fd < 0) {
318*d57664e9SAndroid Build Coastguard Worker goto out;
319*d57664e9SAndroid Build Coastguard Worker }
320*d57664e9SAndroid Build Coastguard Worker success = removeFrom(fd);
321*d57664e9SAndroid Build Coastguard Worker close(fd);
322*d57664e9SAndroid Build Coastguard Worker
323*d57664e9SAndroid Build Coastguard Worker out:
324*d57664e9SAndroid Build Coastguard Worker if (!success) {
325*d57664e9SAndroid Build Coastguard Worker ALOGW("failed to remove signature from %s: %s\n", filename, strerror(errno));
326*d57664e9SAndroid Build Coastguard Worker }
327*d57664e9SAndroid Build Coastguard Worker return success;
328*d57664e9SAndroid Build Coastguard Worker }
329*d57664e9SAndroid Build Coastguard Worker
removeFrom(int fd)330*d57664e9SAndroid Build Coastguard Worker bool ObbFile::removeFrom(int fd)
331*d57664e9SAndroid Build Coastguard Worker {
332*d57664e9SAndroid Build Coastguard Worker if (fd < 0) {
333*d57664e9SAndroid Build Coastguard Worker return false;
334*d57664e9SAndroid Build Coastguard Worker }
335*d57664e9SAndroid Build Coastguard Worker
336*d57664e9SAndroid Build Coastguard Worker if (!readFrom(fd)) {
337*d57664e9SAndroid Build Coastguard Worker return false;
338*d57664e9SAndroid Build Coastguard Worker }
339*d57664e9SAndroid Build Coastguard Worker
340*d57664e9SAndroid Build Coastguard Worker if (ftruncate(fd, mFooterStart) == -1) {
341*d57664e9SAndroid Build Coastguard Worker return false;
342*d57664e9SAndroid Build Coastguard Worker }
343*d57664e9SAndroid Build Coastguard Worker
344*d57664e9SAndroid Build Coastguard Worker return true;
345*d57664e9SAndroid Build Coastguard Worker }
346*d57664e9SAndroid Build Coastguard Worker
347*d57664e9SAndroid Build Coastguard Worker }
348