blob: 963b70ec83353aa6f85a11eccbefe8bb7d139e1a [file] [log] [blame]
Steve Kondik5bd66602016-07-15 10:39:58 -07001#!/bin/bash
2#
3# Copyright (C) 2016 The CyanogenMod Project
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16#
17
18PRODUCT_COPY_FILES_LIST=()
19PRODUCT_COPY_FILES_HASHES=()
20PRODUCT_PACKAGES_LIST=()
21PRODUCT_PACKAGES_HASHES=()
22PACKAGE_LIST=()
23VENDOR_STATE=-1
24VENDOR_RADIO_STATE=-1
25COMMON=-1
26ARCHES=
27FULLY_DEODEXED=-1
28
Rashed Abdel-Tawabe7d9b5c2017-08-05 23:11:35 -040029TMPDIR=$(mktemp -d)
Steve Kondik5bd66602016-07-15 10:39:58 -070030
31#
32# cleanup
33#
34# kill our tmpfiles with fire on exit
35#
36function cleanup() {
37 rm -rf "${TMPDIR:?}"
38}
39
40trap cleanup EXIT INT TERM ERR
41
42#
43# setup_vendor
44#
45# $1: device name
46# $2: vendor name
47# $3: CM root directory
48# $4: is common device - optional, default to false
49# $5: cleanup - optional, default to true
Jake Whatley9843b322017-01-25 21:49:16 -050050# $6: custom vendor makefile name - optional, default to false
Steve Kondik5bd66602016-07-15 10:39:58 -070051#
52# Must be called before any other functions can be used. This
53# sets up the internal state for a new vendor configuration.
54#
55function setup_vendor() {
56 local DEVICE="$1"
57 if [ -z "$DEVICE" ]; then
58 echo "\$DEVICE must be set before including this script!"
59 exit 1
60 fi
61
62 export VENDOR="$2"
63 if [ -z "$VENDOR" ]; then
64 echo "\$VENDOR must be set before including this script!"
65 exit 1
66 fi
67
68 export CM_ROOT="$3"
69 if [ ! -d "$CM_ROOT" ]; then
70 echo "\$CM_ROOT must be set and valid before including this script!"
71 exit 1
72 fi
73
74 export OUTDIR=vendor/"$VENDOR"/"$DEVICE"
75 if [ ! -d "$CM_ROOT/$OUTDIR" ]; then
76 mkdir -p "$CM_ROOT/$OUTDIR"
77 fi
78
Jake Whatley9843b322017-01-25 21:49:16 -050079 VNDNAME="$6"
80 if [ -z "$VNDNAME" ]; then
81 VNDNAME="$DEVICE"
82 fi
83
Jake Whatleycb7cd072016-09-18 20:55:12 +020084 export PRODUCTMK="$CM_ROOT"/"$OUTDIR"/device-vendor.mk
Steve Kondik5bd66602016-07-15 10:39:58 -070085 export ANDROIDMK="$CM_ROOT"/"$OUTDIR"/Android.mk
86 export BOARDMK="$CM_ROOT"/"$OUTDIR"/BoardConfigVendor.mk
87
88 if [ "$4" == "true" ] || [ "$4" == "1" ]; then
89 COMMON=1
90 else
91 COMMON=0
92 fi
93
Gabriele Mc44696d2017-05-01 18:22:04 +020094 if [ "$5" == "false" ] || [ "$5" == "0" ]; then
Steve Kondik5bd66602016-07-15 10:39:58 -070095 VENDOR_STATE=1
96 VENDOR_RADIO_STATE=1
97 else
98 VENDOR_STATE=0
99 VENDOR_RADIO_STATE=0
100 fi
101}
102
103#
104# target_file:
105#
106# $1: colon delimited list
107#
108# Returns destination filename without args
109#
110function target_file() {
111 local LINE="$1"
112 local SPLIT=(${LINE//:/ })
113 local COUNT=${#SPLIT[@]}
114 if [ "$COUNT" -gt "1" ]; then
115 if [[ "${SPLIT[1]}" =~ .*/.* ]]; then
116 printf '%s\n' "${SPLIT[1]}"
117 return 0
118 fi
119 fi
120 printf '%s\n' "${SPLIT[0]}"
121}
122
123#
124# target_args:
125#
Michael Bestas9c6f2eb2018-01-25 21:05:36 +0200126# $1: semicolon delimited list
Steve Kondik5bd66602016-07-15 10:39:58 -0700127#
128# Returns optional arguments (last value) for given target
129#
130function target_args() {
131 local LINE="$1"
Michael Bestas9c6f2eb2018-01-25 21:05:36 +0200132 local SPLIT=(${LINE//;/ })
Steve Kondik5bd66602016-07-15 10:39:58 -0700133 local COUNT=${#SPLIT[@]}
134 if [ "$COUNT" -gt "1" ]; then
135 if [[ ! "${SPLIT[$COUNT-1]}" =~ .*/.* ]]; then
136 printf '%s\n' "${SPLIT[$COUNT-1]}"
137 fi
138 fi
139}
140
141#
142# prefix_match:
143#
144# $1: the prefix to match on
145#
146# Internal function which loops thru the packages list and returns a new
147# list containing the matched files with the prefix stripped away.
148#
149function prefix_match() {
150 local PREFIX="$1"
Vladimir Oltean7220f362018-04-02 22:37:09 +0300151 for LINE in "${PRODUCT_PACKAGES_LIST[@]}"; do
152 local FILE=$(target_file "$LINE")
Steve Kondik5bd66602016-07-15 10:39:58 -0700153 if [[ "$FILE" =~ ^"$PREFIX" ]]; then
154 printf '%s\n' "${FILE#$PREFIX}"
155 fi
156 done
157}
158
159#
Rashed Abdel-Tawab7fd3ccb2017-10-07 14:18:39 -0400160# prefix_match_file:
161#
162# $1: the prefix to match on
163# $2: the file to match the prefix for
164#
165# Internal function which returns true if a filename contains the
166# specified prefix.
167#
168function prefix_match_file() {
169 local PREFIX="$1"
170 local FILE="$2"
171 if [[ "$FILE" =~ ^"$PREFIX" ]]; then
172 return 0
173 else
174 return 1
175 fi
176}
177
178#
179# truncate_file
180#
181# $1: the filename to truncate
182# $2: the argument to output the truncated filename to
183#
184# Internal function which truncates a filename by removing the first dir
185# in the path. ex. vendor/lib/libsdmextension.so -> lib/libsdmextension.so
186#
187function truncate_file() {
188 local FILE="$1"
189 RETURN_FILE="$2"
190 local FIND="${FILE%%/*}"
191 local LOCATION="${#FIND}+1"
192 echo ${FILE:$LOCATION}
193}
194
195#
Steve Kondik5bd66602016-07-15 10:39:58 -0700196# write_product_copy_files:
197#
Rashed Abdel-Tawab7fd3ccb2017-10-07 14:18:39 -0400198# $1: make treble compatible makefile - optional, default to false
199#
Steve Kondik5bd66602016-07-15 10:39:58 -0700200# Creates the PRODUCT_COPY_FILES section in the product makefile for all
201# items in the list which do not start with a dash (-).
202#
203function write_product_copy_files() {
204 local COUNT=${#PRODUCT_COPY_FILES_LIST[@]}
205 local TARGET=
206 local FILE=
207 local LINEEND=
Rashed Abdel-Tawab7fd3ccb2017-10-07 14:18:39 -0400208 local TREBLE_COMPAT=$1
Steve Kondik5bd66602016-07-15 10:39:58 -0700209
210 if [ "$COUNT" -eq "0" ]; then
211 return 0
212 fi
213
214 printf '%s\n' "PRODUCT_COPY_FILES += \\" >> "$PRODUCTMK"
215 for (( i=1; i<COUNT+1; i++ )); do
216 FILE="${PRODUCT_COPY_FILES_LIST[$i-1]}"
217 LINEEND=" \\"
218 if [ "$i" -eq "$COUNT" ]; then
219 LINEEND=""
220 fi
221
222 TARGET=$(target_file "$FILE")
Rashed Abdel-Tawab7fd3ccb2017-10-07 14:18:39 -0400223 if [ "$TREBLE_COMPAT" == "true" ] || [ "$TREBLE_COMPAT" == "1" ]; then
224 if prefix_match_file "vendor/" $TARGET ; then
225 local OUTTARGET=$(truncate_file $TARGET)
226 printf ' %s/proprietary/%s:$(TARGET_COPY_OUT_VENDOR)/%s%s\n' \
227 "$OUTDIR" "$TARGET" "$OUTTARGET" "$LINEEND" >> "$PRODUCTMK"
228 else
229 printf ' %s/proprietary/%s:system/%s%s\n' \
230 "$OUTDIR" "$TARGET" "$TARGET" "$LINEEND" >> "$PRODUCTMK"
231 fi
232 else
233 printf ' %s/proprietary/%s:system/%s%s\n' \
234 "$OUTDIR" "$TARGET" "$TARGET" "$LINEEND" >> "$PRODUCTMK"
235 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700236 done
237 return 0
238}
239
240#
241# write_packages:
242#
243# $1: The LOCAL_MODULE_CLASS for the given module list
244# $2: "true" if this package is part of the vendor/ path
245# $3: type-specific extra flags
246# $4: Name of the array holding the target list
247#
248# Internal function which writes out the BUILD_PREBUILT stanzas
249# for all modules in the list. This is called by write_product_packages
250# after the modules are categorized.
251#
252function write_packages() {
253
254 local CLASS="$1"
255 local VENDOR_PKG="$2"
256 local EXTRA="$3"
257
258 # Yes, this is a horrible hack - we create a new array using indirection
259 local ARR_NAME="$4[@]"
260 local FILELIST=("${!ARR_NAME}")
261
262 local FILE=
263 local ARGS=
264 local BASENAME=
265 local EXTENSION=
266 local PKGNAME=
267 local SRC=
268
269 for P in "${FILELIST[@]}"; do
Michael Bestas9c6f2eb2018-01-25 21:05:36 +0200270 FILE=$(echo $(target_file "$P") | sed 's/\;.*//')
Steve Kondik5bd66602016-07-15 10:39:58 -0700271 ARGS=$(target_args "$P")
272
273 BASENAME=$(basename "$FILE")
M1cha3e8c5bf2017-01-04 09:00:11 +0100274 DIRNAME=$(dirname "$FILE")
Steve Kondik5bd66602016-07-15 10:39:58 -0700275 EXTENSION=${BASENAME##*.}
276 PKGNAME=${BASENAME%.*}
277
278 # Add to final package list
279 PACKAGE_LIST+=("$PKGNAME")
280
281 SRC="proprietary"
282 if [ "$VENDOR_PKG" = "true" ]; then
283 SRC+="/vendor"
284 fi
285
286 printf 'include $(CLEAR_VARS)\n'
287 printf 'LOCAL_MODULE := %s\n' "$PKGNAME"
288 printf 'LOCAL_MODULE_OWNER := %s\n' "$VENDOR"
289 if [ "$CLASS" = "SHARED_LIBRARIES" ]; then
290 if [ "$EXTRA" = "both" ]; then
291 printf 'LOCAL_SRC_FILES_64 := %s/lib64/%s\n' "$SRC" "$FILE"
292 printf 'LOCAL_SRC_FILES_32 := %s/lib/%s\n' "$SRC" "$FILE"
293 #if [ "$VENDOR_PKG" = "true" ]; then
294 # echo "LOCAL_MODULE_PATH_64 := \$(TARGET_OUT_VENDOR_SHARED_LIBRARIES)"
295 # echo "LOCAL_MODULE_PATH_32 := \$(2ND_TARGET_OUT_VENDOR_SHARED_LIBRARIES)"
296 #else
297 # echo "LOCAL_MODULE_PATH_64 := \$(TARGET_OUT_SHARED_LIBRARIES)"
298 # echo "LOCAL_MODULE_PATH_32 := \$(2ND_TARGET_OUT_SHARED_LIBRARIES)"
299 #fi
300 elif [ "$EXTRA" = "64" ]; then
301 printf 'LOCAL_SRC_FILES := %s/lib64/%s\n' "$SRC" "$FILE"
302 else
303 printf 'LOCAL_SRC_FILES := %s/lib/%s\n' "$SRC" "$FILE"
304 fi
305 if [ "$EXTRA" != "none" ]; then
306 printf 'LOCAL_MULTILIB := %s\n' "$EXTRA"
307 fi
308 elif [ "$CLASS" = "APPS" ]; then
Michael Bestas9c6f2eb2018-01-25 21:05:36 +0200309 if [ "$EXTRA" = "priv-app" ]; then
310 SRC="$SRC/priv-app"
311 else
312 SRC="$SRC/app"
Steve Kondik5bd66602016-07-15 10:39:58 -0700313 fi
314 printf 'LOCAL_SRC_FILES := %s/%s\n' "$SRC" "$FILE"
315 local CERT=platform
316 if [ ! -z "$ARGS" ]; then
317 CERT="$ARGS"
318 fi
319 printf 'LOCAL_CERTIFICATE := %s\n' "$CERT"
320 elif [ "$CLASS" = "JAVA_LIBRARIES" ]; then
321 printf 'LOCAL_SRC_FILES := %s/framework/%s\n' "$SRC" "$FILE"
Elektroschmockdd792302016-10-04 21:11:43 +0200322 local CERT=platform
323 if [ ! -z "$ARGS" ]; then
324 CERT="$ARGS"
325 fi
326 printf 'LOCAL_CERTIFICATE := %s\n' "$CERT"
Steve Kondik5bd66602016-07-15 10:39:58 -0700327 elif [ "$CLASS" = "ETC" ]; then
328 printf 'LOCAL_SRC_FILES := %s/etc/%s\n' "$SRC" "$FILE"
329 elif [ "$CLASS" = "EXECUTABLES" ]; then
330 if [ "$ARGS" = "rootfs" ]; then
331 SRC="$SRC/rootfs"
332 if [ "$EXTRA" = "sbin" ]; then
333 SRC="$SRC/sbin"
334 printf '%s\n' "LOCAL_MODULE_PATH := \$(TARGET_ROOT_OUT_SBIN)"
335 printf '%s\n' "LOCAL_UNSTRIPPED_PATH := \$(TARGET_ROOT_OUT_SBIN_UNSTRIPPED)"
336 fi
337 else
338 SRC="$SRC/bin"
339 fi
340 printf 'LOCAL_SRC_FILES := %s/%s\n' "$SRC" "$FILE"
341 unset EXTENSION
342 else
343 printf 'LOCAL_SRC_FILES := %s/%s\n' "$SRC" "$FILE"
344 fi
345 printf 'LOCAL_MODULE_TAGS := optional\n'
346 printf 'LOCAL_MODULE_CLASS := %s\n' "$CLASS"
Hashbang173575f3bb2016-08-28 20:38:45 -0400347 if [ "$CLASS" = "APPS" ]; then
348 printf 'LOCAL_DEX_PREOPT := false\n'
349 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700350 if [ ! -z "$EXTENSION" ]; then
351 printf 'LOCAL_MODULE_SUFFIX := .%s\n' "$EXTENSION"
352 fi
M1cha3e8c5bf2017-01-04 09:00:11 +0100353 if [ "$CLASS" = "SHARED_LIBRARIES" ] || [ "$CLASS" = "EXECUTABLES" ]; then
354 if [ "$DIRNAME" != "." ]; then
355 printf 'LOCAL_MODULE_RELATIVE_PATH := %s\n' "$DIRNAME"
356 fi
357 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700358 if [ "$EXTRA" = "priv-app" ]; then
359 printf 'LOCAL_PRIVILEGED_MODULE := true\n'
360 fi
361 if [ "$VENDOR_PKG" = "true" ]; then
Ethan Chen4f738f52018-02-17 20:03:54 -0800362 printf 'LOCAL_VENDOR_MODULE := true\n'
Steve Kondik5bd66602016-07-15 10:39:58 -0700363 fi
364 printf 'include $(BUILD_PREBUILT)\n\n'
365 done
366}
367
368#
369# write_product_packages:
370#
371# This function will create BUILD_PREBUILT entries in the
372# Android.mk and associated PRODUCT_PACKAGES list in the
373# product makefile for all files in the blob list which
374# start with a single dash (-) character.
375#
376function write_product_packages() {
377 PACKAGE_LIST=()
378
379 local COUNT=${#PRODUCT_PACKAGES_LIST[@]}
380
381 if [ "$COUNT" = "0" ]; then
382 return 0
383 fi
384
385 # Figure out what's 32-bit, what's 64-bit, and what's multilib
386 # I really should not be doing this in bash due to shitty array passing :(
387 local T_LIB32=( $(prefix_match "lib/") )
388 local T_LIB64=( $(prefix_match "lib64/") )
389 local MULTILIBS=( $(comm -12 <(printf '%s\n' "${T_LIB32[@]}") <(printf '%s\n' "${T_LIB64[@]}")) )
390 local LIB32=( $(comm -23 <(printf '%s\n' "${T_LIB32[@]}") <(printf '%s\n' "${MULTILIBS[@]}")) )
391 local LIB64=( $(comm -23 <(printf '%s\n' "${T_LIB64[@]}") <(printf '%s\n' "${MULTILIBS[@]}")) )
392
393 if [ "${#MULTILIBS[@]}" -gt "0" ]; then
394 write_packages "SHARED_LIBRARIES" "false" "both" "MULTILIBS" >> "$ANDROIDMK"
395 fi
396 if [ "${#LIB32[@]}" -gt "0" ]; then
397 write_packages "SHARED_LIBRARIES" "false" "32" "LIB32" >> "$ANDROIDMK"
398 fi
399 if [ "${#LIB64[@]}" -gt "0" ]; then
400 write_packages "SHARED_LIBRARIES" "false" "64" "LIB64" >> "$ANDROIDMK"
401 fi
402
403 local T_V_LIB32=( $(prefix_match "vendor/lib/") )
404 local T_V_LIB64=( $(prefix_match "vendor/lib64/") )
405 local V_MULTILIBS=( $(comm -12 <(printf '%s\n' "${T_V_LIB32[@]}") <(printf '%s\n' "${T_V_LIB64[@]}")) )
406 local V_LIB32=( $(comm -23 <(printf '%s\n' "${T_V_LIB32[@]}") <(printf '%s\n' "${V_MULTILIBS[@]}")) )
407 local V_LIB64=( $(comm -23 <(printf '%s\n' "${T_V_LIB64[@]}") <(printf '%s\n' "${V_MULTILIBS[@]}")) )
408
409 if [ "${#V_MULTILIBS[@]}" -gt "0" ]; then
410 write_packages "SHARED_LIBRARIES" "true" "both" "V_MULTILIBS" >> "$ANDROIDMK"
411 fi
412 if [ "${#V_LIB32[@]}" -gt "0" ]; then
413 write_packages "SHARED_LIBRARIES" "true" "32" "V_LIB32" >> "$ANDROIDMK"
414 fi
415 if [ "${#V_LIB64[@]}" -gt "0" ]; then
416 write_packages "SHARED_LIBRARIES" "true" "64" "V_LIB64" >> "$ANDROIDMK"
417 fi
418
419 # Apps
420 local APPS=( $(prefix_match "app/") )
421 if [ "${#APPS[@]}" -gt "0" ]; then
422 write_packages "APPS" "false" "" "APPS" >> "$ANDROIDMK"
423 fi
424 local PRIV_APPS=( $(prefix_match "priv-app/") )
425 if [ "${#PRIV_APPS[@]}" -gt "0" ]; then
426 write_packages "APPS" "false" "priv-app" "PRIV_APPS" >> "$ANDROIDMK"
427 fi
428 local V_APPS=( $(prefix_match "vendor/app/") )
429 if [ "${#V_APPS[@]}" -gt "0" ]; then
430 write_packages "APPS" "true" "" "V_APPS" >> "$ANDROIDMK"
431 fi
432 local V_PRIV_APPS=( $(prefix_match "vendor/priv-app/") )
433 if [ "${#V_PRIV_APPS[@]}" -gt "0" ]; then
434 write_packages "APPS" "true" "priv-app" "V_PRIV_APPS" >> "$ANDROIDMK"
435 fi
436
437 # Framework
438 local FRAMEWORK=( $(prefix_match "framework/") )
439 if [ "${#FRAMEWORK[@]}" -gt "0" ]; then
440 write_packages "JAVA_LIBRARIES" "false" "" "FRAMEWORK" >> "$ANDROIDMK"
441 fi
Christian Oder974b5902017-10-08 23:15:52 +0200442 local V_FRAMEWORK=( $(prefix_match "vendor/framework/") )
Michael Bestas26eb01e2018-02-27 22:31:55 +0200443 if [ "${#V_FRAMEWORK[@]}" -gt "0" ]; then
Christian Oder974b5902017-10-08 23:15:52 +0200444 write_packages "JAVA_LIBRARIES" "true" "" "V_FRAMEWORK" >> "$ANDROIDMK"
445 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700446
447 # Etc
448 local ETC=( $(prefix_match "etc/") )
449 if [ "${#ETC[@]}" -gt "0" ]; then
450 write_packages "ETC" "false" "" "ETC" >> "$ANDROIDMK"
451 fi
452 local V_ETC=( $(prefix_match "vendor/etc/") )
453 if [ "${#V_ETC[@]}" -gt "0" ]; then
Rashed Abdel-Tawabcc98bc32017-10-08 17:33:42 -0400454 write_packages "ETC" "true" "" "V_ETC" >> "$ANDROIDMK"
Steve Kondik5bd66602016-07-15 10:39:58 -0700455 fi
456
457 # Executables
458 local BIN=( $(prefix_match "bin/") )
459 if [ "${#BIN[@]}" -gt "0" ]; then
460 write_packages "EXECUTABLES" "false" "" "BIN" >> "$ANDROIDMK"
461 fi
462 local V_BIN=( $(prefix_match "vendor/bin/") )
463 if [ "${#V_BIN[@]}" -gt "0" ]; then
464 write_packages "EXECUTABLES" "true" "" "V_BIN" >> "$ANDROIDMK"
465 fi
466 local SBIN=( $(prefix_match "sbin/") )
467 if [ "${#SBIN[@]}" -gt "0" ]; then
468 write_packages "EXECUTABLES" "false" "sbin" "SBIN" >> "$ANDROIDMK"
469 fi
470
471
472 # Actually write out the final PRODUCT_PACKAGES list
473 local PACKAGE_COUNT=${#PACKAGE_LIST[@]}
474
475 if [ "$PACKAGE_COUNT" -eq "0" ]; then
476 return 0
477 fi
478
479 printf '\n%s\n' "PRODUCT_PACKAGES += \\" >> "$PRODUCTMK"
480 for (( i=1; i<PACKAGE_COUNT+1; i++ )); do
481 local LINEEND=" \\"
482 if [ "$i" -eq "$PACKAGE_COUNT" ]; then
483 LINEEND=""
484 fi
485 printf ' %s%s\n' "${PACKAGE_LIST[$i-1]}" "$LINEEND" >> "$PRODUCTMK"
486 done
487}
488
489#
490# write_header:
491#
492# $1: file which will be written to
493#
494# writes out the copyright header with the current year.
495# note that this is not an append operation, and should
496# be executed first!
497#
498function write_header() {
Jake Whatley9843b322017-01-25 21:49:16 -0500499 if [ -f $1 ]; then
500 rm $1
501 fi
502
Steve Kondik5bd66602016-07-15 10:39:58 -0700503 YEAR=$(date +"%Y")
504
505 [ "$COMMON" -eq 1 ] && local DEVICE="$DEVICE_COMMON"
506
Jake Whatley9843b322017-01-25 21:49:16 -0500507 NUM_REGEX='^[0-9]+$'
508 if [[ $INITIAL_COPYRIGHT_YEAR =~ $NUM_REGEX ]] && [ $INITIAL_COPYRIGHT_YEAR -le $YEAR ]; then
509 if [ $INITIAL_COPYRIGHT_YEAR -lt 2016 ]; then
510 printf "# Copyright (C) $INITIAL_COPYRIGHT_YEAR-2016 The CyanogenMod Project\n" > $1
511 elif [ $INITIAL_COPYRIGHT_YEAR -eq 2016 ]; then
512 printf "# Copyright (C) 2016 The CyanogenMod Project\n" > $1
513 fi
514 if [ $YEAR -eq 2017 ]; then
515 printf "# Copyright (C) 2017 The LineageOS Project\n" >> $1
516 elif [ $INITIAL_COPYRIGHT_YEAR -eq $YEAR ]; then
517 printf "# Copyright (C) $YEAR The LineageOS Project\n" >> $1
518 elif [ $INITIAL_COPYRIGHT_YEAR -le 2017 ]; then
519 printf "# Copyright (C) 2017-$YEAR The LineageOS Project\n" >> $1
520 else
521 printf "# Copyright (C) $INITIAL_COPYRIGHT_YEAR-$YEAR The LineageOS Project\n" >> $1
522 fi
523 else
524 printf "# Copyright (C) $YEAR The LineageOS Project\n" > $1
525 fi
526
527 cat << EOF >> $1
Steve Kondik5bd66602016-07-15 10:39:58 -0700528#
529# Licensed under the Apache License, Version 2.0 (the "License");
530# you may not use this file except in compliance with the License.
531# You may obtain a copy of the License at
532#
533# http://www.apache.org/licenses/LICENSE-2.0
534#
535# Unless required by applicable law or agreed to in writing, software
536# distributed under the License is distributed on an "AS IS" BASIS,
537# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
538# See the License for the specific language governing permissions and
539# limitations under the License.
540
541# This file is generated by device/$VENDOR/$DEVICE/setup-makefiles.sh
542
543EOF
544}
545
546#
547# write_headers:
548#
549# $1: devices falling under common to be added to guard - optional
Jake Whatley9843b322017-01-25 21:49:16 -0500550# $2: custom guard - optional
Steve Kondik5bd66602016-07-15 10:39:58 -0700551#
552# Calls write_header for each of the makefiles and creates
553# the initial path declaration and device guard for the
554# Android.mk
555#
556function write_headers() {
557 write_header "$ANDROIDMK"
Jake Whatley9843b322017-01-25 21:49:16 -0500558
559 GUARD="$2"
560 if [ -z "$GUARD" ]; then
561 GUARD="TARGET_DEVICE"
562 fi
563
Steve Kondik5bd66602016-07-15 10:39:58 -0700564 cat << EOF >> "$ANDROIDMK"
565LOCAL_PATH := \$(call my-dir)
566
567EOF
568 if [ "$COMMON" -ne 1 ]; then
569 cat << EOF >> "$ANDROIDMK"
Jake Whatley9843b322017-01-25 21:49:16 -0500570ifeq (\$($GUARD),$DEVICE)
Steve Kondik5bd66602016-07-15 10:39:58 -0700571
572EOF
573 else
574 if [ -z "$1" ]; then
575 echo "Argument with devices to be added to guard must be set!"
576 exit 1
577 fi
578 cat << EOF >> "$ANDROIDMK"
Jake Whatley9843b322017-01-25 21:49:16 -0500579ifneq (\$(filter $1,\$($GUARD)),)
Steve Kondik5bd66602016-07-15 10:39:58 -0700580
581EOF
582 fi
583
584 write_header "$BOARDMK"
585 write_header "$PRODUCTMK"
586}
587
588#
589# write_footers:
590#
591# Closes the inital guard and any other finalization tasks. Must
592# be called as the final step.
593#
594function write_footers() {
595 cat << EOF >> "$ANDROIDMK"
596endif
597EOF
598}
599
600# Return success if adb is up and not in recovery
601function _adb_connected {
602 {
Jake Whatley9843b322017-01-25 21:49:16 -0500603 if [[ "$(adb get-state)" == device ]]
Steve Kondik5bd66602016-07-15 10:39:58 -0700604 then
605 return 0
606 fi
607 } 2>/dev/null
608
609 return 1
610};
611
612#
613# parse_file_list:
614#
615# $1: input file
Rashed Abdel-Tawabb0d08e82017-04-04 02:48:18 -0400616# $2: blob section in file - optional
Steve Kondik5bd66602016-07-15 10:39:58 -0700617#
618# Sets PRODUCT_PACKAGES and PRODUCT_COPY_FILES while parsing the input file
619#
620function parse_file_list() {
621 if [ -z "$1" ]; then
622 echo "An input file is expected!"
623 exit 1
624 elif [ ! -f "$1" ]; then
625 echo "Input file "$1" does not exist!"
626 exit 1
627 fi
628
Rashed Abdel-Tawabb0d08e82017-04-04 02:48:18 -0400629 if [ $# -eq 2 ]; then
630 LIST=$TMPDIR/files.txt
631 cat $1 | sed -n '/# '"$2"'/I,/^\s*$/p' > $LIST
632 else
633 LIST=$1
634 fi
635
636
Steve Kondik5bd66602016-07-15 10:39:58 -0700637 PRODUCT_PACKAGES_LIST=()
638 PRODUCT_PACKAGES_HASHES=()
639 PRODUCT_COPY_FILES_LIST=()
640 PRODUCT_COPY_FILES_HASHES=()
641
642 while read -r line; do
643 if [ -z "$line" ]; then continue; fi
644
645 # If the line has a pipe delimiter, a sha1 hash should follow.
646 # This indicates the file should be pinned and not overwritten
647 # when extracting files.
648 local SPLIT=(${line//\|/ })
649 local COUNT=${#SPLIT[@]}
650 local SPEC=${SPLIT[0]}
651 local HASH="x"
652 if [ "$COUNT" -gt "1" ]; then
653 HASH=${SPLIT[1]}
654 fi
655
656 # if line starts with a dash, it needs to be packaged
657 if [[ "$SPEC" =~ ^- ]]; then
658 PRODUCT_PACKAGES_LIST+=("${SPEC#-}")
659 PRODUCT_PACKAGES_HASHES+=("$HASH")
660 else
661 PRODUCT_COPY_FILES_LIST+=("$SPEC")
662 PRODUCT_COPY_FILES_HASHES+=("$HASH")
663 fi
664
Rashed Abdel-Tawabb0d08e82017-04-04 02:48:18 -0400665 done < <(egrep -v '(^#|^[[:space:]]*$)' "$LIST" | LC_ALL=C sort | uniq)
Steve Kondik5bd66602016-07-15 10:39:58 -0700666}
667
668#
669# write_makefiles:
670#
671# $1: file containing the list of items to extract
Rashed Abdel-Tawab7fd3ccb2017-10-07 14:18:39 -0400672# $2: make treble compatible makefile - optional
Steve Kondik5bd66602016-07-15 10:39:58 -0700673#
674# Calls write_product_copy_files and write_product_packages on
675# the given file and appends to the Android.mk as well as
676# the product makefile.
677#
678function write_makefiles() {
679 parse_file_list "$1"
Rashed Abdel-Tawab7fd3ccb2017-10-07 14:18:39 -0400680 write_product_copy_files "$2"
Steve Kondik5bd66602016-07-15 10:39:58 -0700681 write_product_packages
682}
683
684#
685# append_firmware_calls_to_makefiles:
686#
687# Appends to Android.mk the calls to all images present in radio folder
688# (filesmap file used by releasetools to map firmware images should be kept in the device tree)
689#
690function append_firmware_calls_to_makefiles() {
691 cat << EOF >> "$ANDROIDMK"
692ifeq (\$(LOCAL_PATH)/radio, \$(wildcard \$(LOCAL_PATH)/radio))
693
694RADIO_FILES := \$(wildcard \$(LOCAL_PATH)/radio/*)
695\$(foreach f, \$(notdir \$(RADIO_FILES)), \\
696 \$(call add-radio-file,radio/\$(f)))
697\$(call add-radio-file,../../../device/$VENDOR/$DEVICE/radio/filesmap)
698
699endif
700
701EOF
702}
703
704#
705# get_file:
706#
707# $1: input file
708# $2: target file/folder
709# $3: source of the file (can be "adb" or a local folder)
710#
711# Silently extracts the input file to defined target
712# Returns success if file can be pulled from the device or found locally
713#
714function get_file() {
715 local SRC="$3"
716
717 if [ "$SRC" = "adb" ]; then
718 # try to pull
719 adb pull "$1" "$2" >/dev/null 2>&1 && return 0
720
721 return 1
722 else
723 # try to copy
Jake Whatley9843b322017-01-25 21:49:16 -0500724 cp -r "$SRC/$1" "$2" 2>/dev/null && return 0
Steve Kondik5bd66602016-07-15 10:39:58 -0700725
726 return 1
727 fi
728};
729
730#
731# oat2dex:
732#
733# $1: extracted apk|jar (to check if deodex is required)
734# $2: odexed apk|jar to deodex
735# $3: source of the odexed apk|jar
736#
737# Convert apk|jar .odex in the corresposing classes.dex
738#
739function oat2dex() {
740 local CM_TARGET="$1"
741 local OEM_TARGET="$2"
742 local SRC="$3"
743 local TARGET=
Joe Maplesfb3941c2018-01-05 14:51:33 -0500744 local OAT=
745 local HOST="$(uname)"
Steve Kondik5bd66602016-07-15 10:39:58 -0700746
Joe Maplesfb3941c2018-01-05 14:51:33 -0500747 if [ -z "$ANDROID_HOST_OUT" ]; then
748 echo "ERROR: ANDROID_HOST_OUT not found!"
749 echo "ERROR: Please lunch a device before running this script."
750 exit 1
Steve Kondik5bd66602016-07-15 10:39:58 -0700751 fi
752
Joe Maplesfb3941c2018-01-05 14:51:33 -0500753 if [ -z "$OATDUMP" ] || [ -z "$VDEXEXTRACTOR" ]; then
754 if [ ! -f "$ANDROID_HOST_OUT/bin/oatdump" ]; then
755 echo "ERROR: oatdump utility not found!"
756 echo "ERROR: Please run 'make oatdump'"
757 echo "ERROR: from the top of the android tree before running this script."
758 exit 1
759 else
760 export OATDUMP="$ANDROID_HOST_OUT/bin/oatdump"
761 fi
762 export VDEXEXTRACTOR="$CM_ROOT"/vendor/omni/build/tools/"$HOST"/vdexExtractor
763 fi
764
Steve Kondik5bd66602016-07-15 10:39:58 -0700765 # Extract existing boot.oats to the temp folder
766 if [ -z "$ARCHES" ]; then
Jake Whatley9843b322017-01-25 21:49:16 -0500767 echo "Checking if system is odexed and locating boot.oats..."
Steve Kondik5bd66602016-07-15 10:39:58 -0700768 for ARCH in "arm64" "arm" "x86_64" "x86"; do
Jake Whatley9843b322017-01-25 21:49:16 -0500769 mkdir -p "$TMPDIR/system/framework/$ARCH"
Rashed Abdel-Tawabe7d9b5c2017-08-05 23:11:35 -0400770 if [ -d "$SRC/framework" ] && [ "$SRC" != "adb" ]; then
771 ARCHDIR="framework/$ARCH/"
772 else
773 ARCHDIR="system/framework/$ARCH/"
774 fi
775 if get_file "$ARCHDIR" "$TMPDIR/system/framework/" "$SRC"; then
Steve Kondik5bd66602016-07-15 10:39:58 -0700776 ARCHES+="$ARCH "
Jake Whatley9843b322017-01-25 21:49:16 -0500777 else
778 rmdir "$TMPDIR/system/framework/$ARCH"
Steve Kondik5bd66602016-07-15 10:39:58 -0700779 fi
780 done
781 fi
782
783 if [ -z "$ARCHES" ]; then
784 FULLY_DEODEXED=1 && return 0 # system is fully deodexed, return
785 fi
786
787 if [ ! -f "$CM_TARGET" ]; then
788 return;
789 fi
790
791 if grep "classes.dex" "$CM_TARGET" >/dev/null; then
792 return 0 # target apk|jar is already odexed, return
793 fi
794
795 for ARCH in $ARCHES; do
Jake Whatley9843b322017-01-25 21:49:16 -0500796 BOOTOAT="$TMPDIR/system/framework/$ARCH/boot.oat"
Steve Kondik5bd66602016-07-15 10:39:58 -0700797
Joe Maplesfb3941c2018-01-05 14:51:33 -0500798 local OAT="$(dirname "$OEM_TARGET")/oat/$ARCH/$(basename "$OEM_TARGET" ."${OEM_TARGET##*.}").odex"
799 local VDEX="$(dirname "$OEM_TARGET")/oat/$ARCH/$(basename "$OEM_TARGET" ."${OEM_TARGET##*.}").vdex"
Steve Kondik5bd66602016-07-15 10:39:58 -0700800
Joe Maplesfb3941c2018-01-05 14:51:33 -0500801
802 if get_file "$OAT" "$TMPDIR" "$SRC"; then
803 if get_file "$VDEX" "$TMPDIR" "$SRC"; then
804 "$VDEXEXTRACTOR" -o "$TMPDIR/" -i "$TMPDIR/$(basename "$VDEX")" > /dev/null
805 mv "$TMPDIR/$(basename "${OEM_TARGET%.*}").apk_classes.dex" "$TMPDIR/classes.dex"
806 else
807 "$OATDUMP" --oat-file="$TMPDIR/$(basename "$OAT")" --export-dex-to="$TMPDIR" > /dev/null
808 mv "$(find "$TMPDIR" -maxdepth 1 -type f -name "*_export.dex" | wc -l | tr -d ' ')" "$TMPDIR/classes.dex"
809 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700810 elif [[ "$CM_TARGET" =~ .jar$ ]]; then
Jake Whatley9843b322017-01-25 21:49:16 -0500811 JAROAT="$TMPDIR/system/framework/$ARCH/boot-$(basename ${OEM_TARGET%.*}).oat"
Joe Maplesfb3941c2018-01-05 14:51:33 -0500812 JARVDEX="$TMPDIR/system/framework/$ARCH/boot-$(basename ${OEM_TARGET%.*}).vdex"
Jake Whatley9843b322017-01-25 21:49:16 -0500813 if [ ! -f "$JAROAT" ]; then
814 JAROAT=$BOOTOAT;
815 fi
Joe Maplesfb3941c2018-01-05 14:51:33 -0500816
817 # try to extract classes.dex from boot.vdex for frameworks jars
818 # fallback to boot.oat if vdex is not available
819 if [ -f "$JARVDEX" ]; then
820 "$VDEXEXTRACTOR" -o "$TMPDIR/" -i "$JARVDEX" > /dev/null
821 mv "$TMPDIR/boot-$(basename "${OEM_TARGET%.*}").apk_classes.dex" "$TMPDIR/classes.dex"
822 else
823 "$OATDUMP" --oat-file="$JAROAT" --export-dex-to="$TMPDIR" > /dev/null
824 mv "$(find "$TMPDIR" -maxdepth 1 -type f -name "*_export.dex" | wc -l | tr -d ' ')" "$TMPDIR/classes.dex"
825 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700826 else
827 continue
828 fi
829
Steve Kondik5bd66602016-07-15 10:39:58 -0700830 done
Steve Kondik5bd66602016-07-15 10:39:58 -0700831}
832
833#
834# init_adb_connection:
835#
836# Starts adb server and waits for the device
837#
838function init_adb_connection() {
839 adb start-server # Prevent unexpected starting server message from adb get-state in the next line
840 if ! _adb_connected; then
841 echo "No device is online. Waiting for one..."
842 echo "Please connect USB and/or enable USB debugging"
843 until _adb_connected; do
844 sleep 1
845 done
846 echo "Device Found."
847 fi
848
849 # Retrieve IP and PORT info if we're using a TCP connection
850 TCPIPPORT=$(adb devices | egrep '^[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+:[0-9]+[^0-9]+' \
851 | head -1 | awk '{print $1}')
852 adb root &> /dev/null
853 sleep 0.3
854 if [ -n "$TCPIPPORT" ]; then
855 # adb root just killed our connection
856 # so reconnect...
857 adb connect "$TCPIPPORT"
858 fi
859 adb wait-for-device &> /dev/null
860 sleep 0.3
861}
862
863#
864# fix_xml:
865#
866# $1: xml file to fix
867#
868function fix_xml() {
869 local XML="$1"
870 local TEMP_XML="$TMPDIR/`basename "$XML"`.temp"
871
Dobroslaw Kijowski3af2a8d2017-05-18 12:35:02 +0200872 grep -a '^<?xml version' "$XML" > "$TEMP_XML"
873 grep -av '^<?xml version' "$XML" >> "$TEMP_XML"
Steve Kondik5bd66602016-07-15 10:39:58 -0700874
875 mv "$TEMP_XML" "$XML"
876}
877
878#
879# extract:
880#
881# $1: file containing the list of items to extract
Dan Pasanen0cc05012017-03-21 09:06:11 -0500882# $2: path to extracted system folder, an ota zip file, or "adb" to extract from device
Rashed Abdel-Tawabb0d08e82017-04-04 02:48:18 -0400883# $3: section in list file to extract - optional
Steve Kondik5bd66602016-07-15 10:39:58 -0700884#
885function extract() {
886 if [ -z "$OUTDIR" ]; then
887 echo "Output dir not set!"
888 exit 1
889 fi
890
Harry Youd972c4112017-08-05 09:18:56 +0100891 if [ -z "$3" ]; then
892 parse_file_list "$1"
893 else
894 parse_file_list "$1" "$3"
895 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700896
897 # Allow failing, so we can try $DEST and/or $FILE
898 set +e
899
900 local FILELIST=( ${PRODUCT_COPY_FILES_LIST[@]} ${PRODUCT_PACKAGES_LIST[@]} )
901 local HASHLIST=( ${PRODUCT_COPY_FILES_HASHES[@]} ${PRODUCT_PACKAGES_HASHES[@]} )
902 local COUNT=${#FILELIST[@]}
903 local SRC="$2"
904 local OUTPUT_ROOT="$CM_ROOT"/"$OUTDIR"/proprietary
905 local OUTPUT_TMP="$TMPDIR"/"$OUTDIR"/proprietary
906
907 if [ "$SRC" = "adb" ]; then
908 init_adb_connection
909 fi
910
Dan Pasanen0cc05012017-03-21 09:06:11 -0500911 if [ -f "$SRC" ] && [ "${SRC##*.}" == "zip" ]; then
912 DUMPDIR="$CM_ROOT"/system_dump
913
914 # Check if we're working with the same zip that was passed last time.
915 # If so, let's just use what's already extracted.
916 MD5=`md5sum "$SRC"| awk '{print $1}'`
917 OLDMD5=`cat "$DUMPDIR"/zipmd5.txt`
918
919 if [ "$MD5" != "$OLDMD5" ]; then
920 rm -rf "$DUMPDIR"
921 mkdir "$DUMPDIR"
922 unzip "$SRC" -d "$DUMPDIR"
923 echo "$MD5" > "$DUMPDIR"/zipmd5.txt
924
925 # Stop if an A/B OTA zip is detected. We cannot extract these.
926 if [ -a "$DUMPDIR"/payload.bin ]; then
927 echo "A/B style OTA zip detected. This is not supported at this time. Stopping..."
928 exit 1
929 # If OTA is block based, extract it.
930 elif [ -a "$DUMPDIR"/system.new.dat ]; then
931 echo "Converting system.new.dat to system.img"
932 python "$CM_ROOT"/vendor/omni/build/tools/sdat2img.py "$DUMPDIR"/system.transfer.list "$DUMPDIR"/system.new.dat "$DUMPDIR"/system.img 2>&1
933 rm -rf "$DUMPDIR"/system.new.dat "$DUMPDIR"/system
934 mkdir "$DUMPDIR"/system "$DUMPDIR"/tmp
935 echo "Requesting sudo access to mount the system.img"
936 sudo mount -o loop "$DUMPDIR"/system.img "$DUMPDIR"/tmp
937 cp -r "$DUMPDIR"/tmp/* "$DUMPDIR"/system/
938 sudo umount "$DUMPDIR"/tmp
939 rm -rf "$DUMPDIR"/tmp "$DUMPDIR"/system.img
940 fi
941 fi
942
943 SRC="$DUMPDIR"
944 fi
945
Steve Kondik5bd66602016-07-15 10:39:58 -0700946 if [ "$VENDOR_STATE" -eq "0" ]; then
947 echo "Cleaning output directory ($OUTPUT_ROOT).."
948 rm -rf "${OUTPUT_TMP:?}"
949 mkdir -p "${OUTPUT_TMP:?}"
Jake Whatley9843b322017-01-25 21:49:16 -0500950 if [ -d "$OUTPUT_ROOT" ]; then
951 mv "${OUTPUT_ROOT:?}/"* "${OUTPUT_TMP:?}/"
952 fi
Steve Kondik5bd66602016-07-15 10:39:58 -0700953 VENDOR_STATE=1
954 fi
955
956 echo "Extracting $COUNT files in $1 from $SRC:"
957
958 for (( i=1; i<COUNT+1; i++ )); do
959
960 local FROM=$(target_file "${FILELIST[$i-1]}")
961 local ARGS=$(target_args "${FILELIST[$i-1]}")
962 local SPLIT=(${FILELIST[$i-1]//:/ })
Michael Bestas9c6f2eb2018-01-25 21:05:36 +0200963 local FILE=$(echo "${SPLIT[0]#-}" | sed 's/\;.*//')
Steve Kondik5bd66602016-07-15 10:39:58 -0700964 local OUTPUT_DIR="$OUTPUT_ROOT"
965 local TMP_DIR="$OUTPUT_TMP"
966 local TARGET=
967
968 if [ "$ARGS" = "rootfs" ]; then
969 TARGET="$FROM"
970 OUTPUT_DIR="$OUTPUT_DIR/rootfs"
971 TMP_DIR="$TMP_DIR/rootfs"
Rashed Abdel-Tawab7da4a1b2017-04-04 18:03:35 -0400972 elif [ -f "$SRC/$FILE" ] && [ "$SRC" != "adb" ]; then
973 TARGET="$FROM"
Steve Kondik5bd66602016-07-15 10:39:58 -0700974 else
975 TARGET="system/$FROM"
976 FILE="system/$FILE"
977 fi
978
979 if [ "$SRC" = "adb" ]; then
980 printf ' - %s .. ' "/$TARGET"
981 else
982 printf ' - %s \n' "/$TARGET"
983 fi
984
985 local DIR=$(dirname "$FROM")
986 if [ ! -d "$OUTPUT_DIR/$DIR" ]; then
987 mkdir -p "$OUTPUT_DIR/$DIR"
988 fi
989 local DEST="$OUTPUT_DIR/$FROM"
990
Gabriele M58270a32017-11-13 23:15:29 +0100991 # Check pinned files
992 local HASH="${HASHLIST[$i-1]}"
993 local KEEP=""
994 if [ "$DISABLE_PINNING" != "1" ] && [ ! -z "$HASH" ] && [ "$HASH" != "x" ]; then
995 if [ -f "$DEST" ]; then
996 local PINNED="$DEST"
997 else
998 local PINNED="$TMP_DIR/$FROM"
999 fi
1000 if [ -f "$PINNED" ]; then
1001 if [ "$(uname)" == "Darwin" ]; then
1002 local TMP_HASH=$(shasum "$PINNED" | awk '{print $1}' )
1003 else
1004 local TMP_HASH=$(sha1sum "$PINNED" | awk '{print $1}' )
1005 fi
1006 if [ "$TMP_HASH" = "$HASH" ]; then
1007 KEEP="1"
1008 if [ ! -f "$DEST" ]; then
1009 cp -p "$PINNED" "$DEST"
1010 fi
1011 fi
1012 fi
1013 fi
1014
1015 if [ "$KEEP" = "1" ]; then
1016 printf ' + (keeping pinned file with hash %s)\n' "$HASH"
1017 elif [ "$SRC" = "adb" ]; then
Steve Kondik5bd66602016-07-15 10:39:58 -07001018 # Try CM target first
1019 adb pull "/$TARGET" "$DEST"
1020 # if file does not exist try OEM target
1021 if [ "$?" != "0" ]; then
1022 adb pull "/$FILE" "$DEST"
1023 fi
1024 else
Christopher R. Palmer1fbf6872017-03-04 05:12:29 -05001025 # Try CM target first
1026 if [ -f "$SRC/$TARGET" ]; then
Steve Kondik5bd66602016-07-15 10:39:58 -07001027 cp "$SRC/$TARGET" "$DEST"
Christopher R. Palmer1fbf6872017-03-04 05:12:29 -05001028 # if file does not exist try OEM target
1029 elif [ -f "$SRC/$FILE" ]; then
1030 cp "$SRC/$FILE" "$DEST"
Steve Kondik5bd66602016-07-15 10:39:58 -07001031 else
1032 printf ' !! file not found in source\n'
1033 fi
1034 fi
1035
1036 if [ "$?" == "0" ]; then
1037 # Deodex apk|jar if that's the case
1038 if [[ "$FULLY_DEODEXED" -ne "1" && "$DEST" =~ .(apk|jar)$ ]]; then
1039 oat2dex "$DEST" "$FILE" "$SRC"
1040 if [ -f "$TMPDIR/classes.dex" ]; then
1041 zip -gjq "$DEST" "$TMPDIR/classes.dex"
1042 rm "$TMPDIR/classes.dex"
1043 printf ' (updated %s from odex files)\n' "/$FILE"
1044 fi
1045 elif [[ "$DEST" =~ .xml$ ]]; then
1046 fix_xml "$DEST"
1047 fi
1048 fi
1049
1050 # Check pinned files
1051 local HASH="${HASHLIST[$i-1]}"
Jake Whatley9843b322017-01-25 21:49:16 -05001052 if [ "$DISABLE_PINNING" != "1" ] && [ ! -z "$HASH" ] && [ "$HASH" != "x" ]; then
Steve Kondik5bd66602016-07-15 10:39:58 -07001053 local KEEP=""
1054 local TMP="$TMP_DIR/$FROM"
1055 if [ -f "$TMP" ]; then
1056 if [ ! -f "$DEST" ]; then
1057 KEEP="1"
1058 else
Jake Whatley9843b322017-01-25 21:49:16 -05001059 if [ "$(uname)" == "Darwin" ]; then
1060 local DEST_HASH=$(shasum "$DEST" | awk '{print $1}' )
1061 else
1062 local DEST_HASH=$(sha1sum "$DEST" | awk '{print $1}' )
1063 fi
Steve Kondik5bd66602016-07-15 10:39:58 -07001064 if [ "$DEST_HASH" != "$HASH" ]; then
1065 KEEP="1"
1066 fi
1067 fi
1068 if [ "$KEEP" = "1" ]; then
Jake Whatley9843b322017-01-25 21:49:16 -05001069 if [ "$(uname)" == "Darwin" ]; then
1070 local TMP_HASH=$(shasum "$TMP" | awk '{print $1}' )
1071 else
1072 local TMP_HASH=$(sha1sum "$TMP" | awk '{print $1}' )
1073 fi
Steve Kondik5bd66602016-07-15 10:39:58 -07001074 if [ "$TMP_HASH" = "$HASH" ]; then
1075 printf ' + (keeping pinned file with hash %s)\n' "$HASH"
1076 cp -p "$TMP" "$DEST"
1077 fi
1078 fi
1079 fi
1080 fi
1081
1082 if [ -f "$DEST" ]; then
1083 local TYPE="${DIR##*/}"
1084 if [ "$TYPE" = "bin" -o "$TYPE" = "sbin" ]; then
1085 chmod 755 "$DEST"
1086 else
1087 chmod 644 "$DEST"
1088 fi
1089 fi
1090
1091 done
1092
1093 # Don't allow failing
1094 set -e
1095}
1096
1097#
1098# extract_firmware:
1099#
1100# $1: file containing the list of items to extract
1101# $2: path to extracted radio folder
1102#
1103function extract_firmware() {
1104 if [ -z "$OUTDIR" ]; then
1105 echo "Output dir not set!"
1106 exit 1
1107 fi
1108
1109 parse_file_list "$1"
1110
1111 # Don't allow failing
1112 set -e
1113
1114 local FILELIST=( ${PRODUCT_COPY_FILES_LIST[@]} )
1115 local COUNT=${#FILELIST[@]}
1116 local SRC="$2"
1117 local OUTPUT_DIR="$CM_ROOT"/"$OUTDIR"/radio
1118
1119 if [ "$VENDOR_RADIO_STATE" -eq "0" ]; then
1120 echo "Cleaning firmware output directory ($OUTPUT_DIR).."
1121 rm -rf "${OUTPUT_DIR:?}/"*
1122 VENDOR_RADIO_STATE=1
1123 fi
1124
1125 echo "Extracting $COUNT files in $1 from $SRC:"
1126
1127 for (( i=1; i<COUNT+1; i++ )); do
1128 local FILE="${FILELIST[$i-1]}"
1129 printf ' - %s \n' "/radio/$FILE"
1130
1131 if [ ! -d "$OUTPUT_DIR" ]; then
1132 mkdir -p "$OUTPUT_DIR"
1133 fi
1134 cp "$SRC/$FILE" "$OUTPUT_DIR/$FILE"
1135 chmod 644 "$OUTPUT_DIR/$FILE"
1136 done
1137}