- add sources.
[platform/framework/web/crosswalk.git] / src / tools / resources / optimize-png-files.sh
1 #!/bin/bash
2 # Copyright 2013 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file.
5
6 # The optimization code is based on pngslim (http://goo.gl/a0XHg)
7 # and executes a similar pipleline to optimize the png file size.
8 # The steps that require pngoptimizercl/pngrewrite/deflopt are omitted,
9 # but this runs all other processes, including:
10 # 1) various color-dependent optimizations using optipng.
11 # 2) optimize the number of huffman blocks.
12 # 3) randomize the huffman table.
13 # 4) Further optimize using optipng and advdef (zlib stream).
14 # Due to the step 3), each run may produce slightly different results.
15 #
16 # Note(oshima): In my experiment, advdef didn't reduce much. I'm keeping it
17 # for now as it does not take much time to run.
18
19 readonly ALL_DIRS="
20 ash/resources
21 ui/resources
22 chrome/app/theme
23 chrome/browser/resources
24 chrome/renderer/resources
25 webkit/glue/resources
26 remoting/resources
27 remoting/webapp
28 "
29
30 # Files larger than this file size (in bytes) will
31 # use the optimization parameters tailored for large files.
32 LARGE_FILE_THRESHOLD=3000
33
34 # Constants used for optimization
35 readonly DEFAULT_MIN_BLOCK_SIZE=128
36 readonly DEFAULT_LIMIT_BLOCKS=256
37 readonly DEFAULT_RANDOM_TRIALS=100
38 # Taken from the recommendation in the pngslim's readme.txt.
39 readonly LARGE_MIN_BLOCK_SIZE=1
40 readonly LARGE_LIMIT_BLOCKS=2
41 readonly LARGE_RANDOM_TRIALS=1
42
43 # Global variables for stats
44 TOTAL_OLD_BYTES=0
45 TOTAL_NEW_BYTES=0
46 TOTAL_FILE=0
47 PROCESSED_FILE=0
48
49 declare -a THROBBER_STR=('-' '\\' '|' '/')
50 THROBBER_COUNT=0
51
52 # Show throbber character at current cursor position.
53 function throbber {
54   echo -ne "${THROBBER_STR[$THROBBER_COUNT]}\b"
55   let THROBBER_COUNT=($THROBBER_COUNT+1)%4
56 }
57
58 # Usage: pngout_loop <file> <png_out_options> ...
59 # Optimize the png file using pngout with the given options
60 # using various block split thresholds and filter types.
61 function pngout_loop {
62   local file=$1
63   shift
64   local opts=$*
65   if [ $OPTIMIZE_LEVEL == 1 ]; then
66     for j in $(seq 0 5); do
67       throbber
68       pngout -q -k1 -s1 -f$j $opts $file
69     done
70   else
71     for i in 0 128 256 512; do
72       for j in $(seq 0 5); do
73         throbber
74         pngout -q -k1 -s1 -b$i -f$j $opts $file
75       done
76     done
77   fi
78 }
79
80 # Usage: get_color_depth_list
81 # Returns the list of color depth options for current optimization level.
82 function get_color_depth_list {
83   if [ $OPTIMIZE_LEVEL == 1 ]; then
84     echo "-d0"
85   else
86     echo "-d1 -d2 -d4 -d8"
87   fi
88 }
89
90 # Usage: process_grayscale <file>
91 # Optimize grayscale images for all color bit depths.
92 #
93 # TODO(oshima): Experiment with -d0 w/o -c0.
94 function process_grayscale {
95   echo -n "|gray"
96   for opt in $(get_color_depth_list); do
97     pngout_loop $file -c0 $opt
98   done
99 }
100
101 # Usage: process_grayscale_alpha <file>
102 # Optimize grayscale images with alpha for all color bit depths.
103 function process_grayscale_alpha {
104   echo -n "|gray-a"
105   pngout_loop $file -c4
106   for opt in $(get_color_depth_list); do
107     pngout_loop $file -c3 $opt
108   done
109 }
110
111 # Usage: process_rgb <file>
112 # Optimize rgb images with or without alpha for all color bit depths.
113 function process_rgb {
114   echo -n "|rgb"
115   for opt in $(get_color_depth_list); do
116     pngout_loop $file -c3 $opt
117   done
118   pngout_loop $file -c2
119   pngout_loop $file -c6
120 }
121
122 # Usage: huffman_blocks <file>
123 # Optimize the huffman blocks.
124 function huffman_blocks {
125   local file=$1
126   echo -n "|huffman"
127   local size=$(stat -c%s $file)
128   local min_block_size=$DEFAULT_MIN_BLOCK_SIZE
129   local limit_blocks=$DEFAULT_LIMIT_BLOCKS
130
131   if [ $size -gt $LARGE_FILE_THRESHOLD ]; then
132     min_block_size=$LARGE_MIN_BLOCK_SIZE
133     limit_blocks=$LARGE_LIMIT_BLOCKS
134   fi
135   let max_blocks=$size/$min_block_size
136   if [ $max_blocks -gt $limit_blocks ]; then
137     max_blocks=$limit_blocks
138   fi
139
140   for i in $(seq 2 $max_blocks); do
141     throbber
142     pngout -q -k1 -ks -s1 -n$i $file
143   done
144 }
145
146 # Usage: random_huffman_table_trial <file>
147 # Try compressing by randomizing the initial huffman table.
148 #
149 # TODO(oshima): Try adjusting different parameters for large files to
150 # reduce runtime.
151 function random_huffman_table_trial {
152   echo -n "|random"
153   local file=$1
154   local old_size=$(stat -c%s $file)
155   local trials_count=$DEFAULT_RANDOM_TRIALS
156
157   if [ $old_size -gt $LARGE_FILE_THRESHOLD ]; then
158     trials_count=$LARGE_RANDOM_TRIALS
159   fi
160   for i in $(seq 1 $trials_count); do
161     throbber
162     pngout -q -k1 -ks -s0 -r $file
163   done
164   local new_size=$(stat -c%s $file)
165   if [ $new_size -lt $old_size ]; then
166     random_huffman_table_trial $file
167   fi
168 }
169
170 # Usage: final_comprssion <file>
171 # Further compress using optipng and advdef.
172 # TODO(oshima): Experiment with 256.
173 function final_compression {
174   echo -n "|final"
175   local file=$1
176   if [ $OPTIMIZE_LEVEL == 2 ]; then
177     for i in 32k 16k 8k 4k 2k 1k 512; do
178       throbber
179       optipng -q -nb -nc -zw$i -zc1-9 -zm1-9 -zs0-3 -f0-5 $file
180     done
181   fi
182   for i in $(seq 1 4); do
183     throbber
184     advdef -q -z -$i $file
185   done
186   echo -ne "\r"
187 }
188
189 # Usage: get_color_type <file>
190 # Returns the color type name of the png file. Here is the list of names
191 # for each color type codes.
192 # 0 : grayscale
193 # 2 : RGB
194 # 3 : colormap
195 # 4 : gray+alpha
196 # 6 : RGBA
197 # See http://en.wikipedia.org/wiki/Portable_Network_Graphics#Color_depth
198 # for details about the color type code.
199 function get_color_type {
200   local file=$1
201   echo $(file $file | awk -F, '{print $3}' | awk '{print $2}')
202 }
203
204 # Usage: optimize_size <file>
205 # Performs png file optimization.
206 function optimize_size {
207   tput el
208   local file=$1
209   echo -n "$file "
210
211   advdef -q -z -4 $file
212
213   pngout -q -s4 -c0 -force $file $file.tmp.png
214   if [ -f $file.tmp.png ]; then
215     rm $file.tmp.png
216     process_grayscale $file
217     process_grayscale_alpha $file
218   else
219     pngout -q -s4 -c4 -force $file $file.tmp.png
220     if [ -f $file.tmp.png ]; then
221       rm $file.tmp.png
222       process_grayscale_alpha $file
223     else
224       process_rgb $file
225     fi
226   fi
227
228   echo -n "|filter"
229   local old_color_type=$(get_color_type $file)
230   optipng -q -zc9 -zm8 -zs0-3 -f0-5 $file -out $file.tmp.png
231   local new_color_type=$(get_color_type $file.tmp.png)
232   # optipng may corrupt a png file when reducing the color type
233   # to grayscale/grayscale+alpha. Just skip such cases until
234   # the bug is fixed. See crbug.com/174505, crbug.com/174084.
235   # The issue is reported in
236   # https://sourceforge.net/tracker/?func=detail&aid=3603630&group_id=151404&atid=780913
237   if [[ $old_color_type == "RGBA" && $new_color_type =~ gray.* ]] ; then
238     rm $file.tmp.png
239     echo -n "[skip opting]"
240   else
241     mv $file.tmp.png $file
242   fi
243   pngout -q -k1 -s1 $file
244
245   huffman_blocks $file
246
247   # TODO(oshima): Experiment with strategy 1.
248   echo -n "|strategy"
249   if [ $OPTIMIZE_LEVEL == 2 ]; then
250     for i in 3 2 0; do
251       pngout -q -k1 -ks -s$i $file
252     done
253   else
254     pngout -q -k1 -ks -s1 $file
255   fi
256
257   if [ $OPTIMIZE_LEVEL == 2 ]; then
258     random_huffman_table_trial $file
259   fi
260
261   final_compression $file
262 }
263
264 # Usage: process_file <file>
265 function process_file {
266   local file=$1
267   local name=$(basename $file)
268   # -rem alla removes all ancillary chunks except for tRNS
269   pngcrush -d $TMP_DIR -brute -reduce -rem alla $file > /dev/null
270
271   if [ $OPTIMIZE_LEVEL != 0 ]; then
272     optimize_size $TMP_DIR/$name
273   fi
274 }
275
276 # Usage: optimize_file <file>
277 function optimize_file {
278   local file=$1
279   if $using_cygwin ; then
280     file=$(cygpath -w $file)
281   fi
282
283   local name=$(basename $file)
284   local old=$(stat -c%s $file)
285   local tmp_file=$TMP_DIR/$name
286
287   process_file $file
288
289   local new=$(stat -c%s $tmp_file)
290   let diff=$old-$new
291   let percent=($diff*100)/$old
292   let TOTAL_FILE+=1
293
294   tput el
295   if [ $new -lt $old ]; then
296     echo -ne "$file : $old => $new ($diff bytes : $percent %)\n"
297     mv "$tmp_file" "$file"
298     let TOTAL_OLD_BYTES+=$old
299     let TOTAL_NEW_BYTES+=$new
300     let PROCESSED_FILE+=1
301   else
302     if [ $OPTIMIZE_LEVEL == 0 ]; then
303       echo -ne "$file : skipped\r"
304     fi
305     rm $tmp_file
306   fi
307 }
308
309 function optimize_dir {
310   local dir=$1
311   if $using_cygwin ; then
312     dir=$(cygpath -w $dir)
313   fi
314
315   for f in $(find $dir -name "*.png"); do
316     optimize_file $f
317   done
318 }
319
320 function install_if_not_installed {
321   local program=$1
322   local package=$2
323   which $program > /dev/null 2>&1
324   if [ "$?" != "0" ]; then
325     if $using_cygwin ; then
326       echo "Couldn't find $program. " \
327            "Please run cygwin's setup.exe and install the $package package."
328       exit 1
329     else
330       read -p "Couldn't find $program. Do you want to install? (y/n)"
331       [ "$REPLY" == "y" ] && sudo apt-get install $package
332       [ "$REPLY" == "y" ] || exit
333     fi
334   fi
335 }
336
337 function fail_if_not_installed {
338   local program=$1
339   local url=$2
340   which $program > /dev/null 2>&1
341   if [ $? != 0 ]; then
342     echo "Couldn't find $program. Please download and install it from $url ."
343     exit 1
344   fi
345 }
346
347 function show_help {
348   local program=$(basename $0)
349   echo \
350 "Usage: $program [options] dir ...
351
352 $program is a utility to reduce the size of png files by removing
353 unnecessary chunks and compressing the image.
354
355 Options:
356   -o<optimize_level>  Specify optimization level: (default is 1)
357       0  Just run pngcrush. It removes unnecessary chunks and perform basic
358          optimization on the encoded data.
359       1  Optimize png files using pngout/optipng and advdef. This can further
360          reduce addtional 5~30%. This is the default level.
361       2  Aggressively optimize the size of png files. This may produce
362          addtional 1%~5% reduction.  Warning: this is *VERY*
363          slow and can take hours to process all files.
364   -r<revision> If this is specified, the script processes only png files
365                changed since this revision. The <dir> options will be used
366                to narrow down the files under specific directories.
367   -h  Print this help text."
368   exit 1
369 }
370
371 if [ ! -e ../.gclient ]; then
372   echo "$0 must be run in src directory"
373   exit 1
374 fi
375
376 if [ "$(expr substr $(uname -s) 1 6)" == "CYGWIN" ]; then
377   using_cygwin=true
378 else
379   using_cygwin=false
380 fi
381
382 OPTIMIZE_LEVEL=1
383 # Parse options
384 while getopts o:r:h opts
385 do
386   case $opts in
387     r)
388       COMMIT=$(git svn find-rev r$OPTARG | tail -1) || exit
389       if [ -z "$COMMIT" ] ; then
390         echo "Revision $OPTARG not found"
391         show_help
392       fi
393       ;;
394     o)
395       if [[ ! "$OPTARG" =~ [012] ]] ; then
396         show_help
397       fi
398       OPTIMIZE_LEVEL=$OPTARG
399       ;;
400     [h?])
401       show_help;;
402   esac
403 done
404
405 # Remove options from argument list.
406 shift $(($OPTIND -1))
407
408 # Make sure we have all necessary commands installed.
409 install_if_not_installed pngcrush pngcrush
410 if [ $OPTIMIZE_LEVEL -ge 1 ]; then
411   install_if_not_installed optipng optipng
412
413   if $using_cygwin ; then
414     fail_if_not_installed advdef "http://advancemame.sourceforge.net/comp-readme.html"
415   else
416     install_if_not_installed advdef advancecomp
417   fi
418
419   if $using_cygwin ; then
420     pngout_url="http://www.advsys.net/ken/utils.htm"
421   else
422     pngout_url="http://www.jonof.id.au/kenutils"
423   fi
424   fail_if_not_installed pngout $pngout_url
425 fi
426
427 # Create tmp directory for crushed png file.
428 TMP_DIR=$(mktemp -d)
429 if $using_cygwin ; then
430   TMP_DIR=$(cygpath -w $TMP_DIR)
431 fi
432
433 # Make sure we cleanup temp dir
434 trap "rm -rf $TMP_DIR" EXIT
435
436 # If no directories are specified, optimize all directories.
437 DIRS=$@
438 set ${DIRS:=$ALL_DIRS}
439
440 echo "Optimize level=$OPTIMIZE_LEVEL"
441 if [ -n "$COMMIT" ] ; then
442  ALL_FILES=$(git diff --name-only $COMMIT HEAD $DIRS | grep "png$")
443  ALL_FILES_LIST=( $ALL_FILES )
444  echo "Processing ${#ALL_FILES_LIST[*]} files"
445  for f in $ALL_FILES; do
446    if [ -f $f ] ; then
447      optimize_file $f
448    else
449      echo "Skipping deleted file: $f";
450    fi
451  done
452 else
453   for d in $DIRS; do
454     echo "Optimizing png files in $d"
455     optimize_dir $d
456     echo
457   done
458 fi
459
460 # Print the results.
461 if [ $PROCESSED_FILE == 0 ]; then
462   echo "Did not find any files (out of $TOTAL_FILE files)" \
463        "that could be optimized" \
464        "in $(date -u -d @$SECONDS +%T)s"
465 else
466   let diff=$TOTAL_OLD_BYTES-$TOTAL_NEW_BYTES
467   let percent=$diff*100/$TOTAL_OLD_BYTES
468   echo "Processed $PROCESSED_FILE files (out of $TOTAL_FILE files)" \
469        "in $(date -u -d @$SECONDS +%T)s"
470   echo "Result : $TOTAL_OLD_BYTES => $TOTAL_NEW_BYTES bytes" \
471        "($diff bytes : $percent %)"
472 fi