इस दस्तावेज़ में, पार्टनर को कुछ खास Android डिवाइसों के लिए, बूट होने में लगने वाले समय को बेहतर बनाने के बारे में बताया गया है. बूट होने में लगने वाला समय, सिस्टम की परफ़ॉर्मेंस का एक अहम कॉम्पोनेंट है. डिवाइस का इस्तेमाल करने से पहले, उपयोगकर्ताओं को बूट होने का इंतज़ार करना पड़ता है. कार जैसे डिवाइसों के लिए, तेज़ी से चालू होने की सुविधा ज़रूरी है. इन डिवाइसों में अक्सर कोल्ड बूट-अप होता है. किसी नेविगेशन डेस्टिनेशन को डालने के लिए, ज़्यादा सेकंड तक इंतज़ार करना किसी को पसंद नहीं आता.
Android 8.0 में कई कॉम्पोनेंट में सुधार किए गए हैं. इससे, डिवाइस को बूट होने में कम समय लगता है. यहां दी गई टेबल में, परफ़ॉर्मेंस में हुए इन सुधारों के बारे में खास जानकारी दी गई है. ये सुधार, Google Pixel और Pixel XL डिवाइसों पर किए गए हैं.
| कॉम्पोनेंट | सुधार |
|---|---|
| बूटलोडर |
|
| डिवाइस का kernel |
|
| I/O ट्यूनिंग |
|
| init.*.rc |
|
| बूट ऐनिमेशन |
|
| SELinux नीति | genfscon की मदद से 0.2 सेकंड बचाए गए |
बूटलोडर को ऑप्टिमाइज़ करना
बूट होने में लगने वाले समय को कम करने के लिए, बूटलोडर को ऑप्टिमाइज़ करने के लिए:
- लॉग करने के लिए:
- UART में लॉग लिखने की सुविधा बंद करें, क्योंकि बहुत ज़्यादा लॉगिंग में काफ़ी समय लग सकता है. (हमें पता चला है कि Google Pixel डिवाइसों पर, यह बूटलोडर को 1.5 सेकंड तक धीमा कर देता है).
- सिर्फ़ गड़बड़ी की स्थितियों को लॉग करें और अन्य जानकारी को मेमोरी में सेव करें. साथ ही, उसे वापस पाने के लिए अलग से कोई तरीका अपनाएं.
- कर्नेल को डिकंप्रेस करने के लिए, GZIP के बजाय, ज़्यादा बेहतर हार्डवेयर के लिए LZ4 का इस्तेमाल करें (उदाहरण के लिए, पैच). ध्यान रखें कि अलग-अलग कर्नेल कंप्रेसन विकल्पों के लोड होने और डिकंप्रेस होने में अलग-अलग समय लग सकता है. साथ ही, हो सकता है कि आपके खास हार्डवेयर के लिए, कुछ विकल्प दूसरों के मुकाबले बेहतर काम करें.
- डिबाउंसिंग/खास मोड में एंट्री के लिए, ग़ैर-ज़रूरी इंतज़ार का समय देखें और उसे कम करें.
- बूटलोडर में बूट होने में लगने वाले समय को, cmdline के तौर पर कर्नेल को पास करें.
- सीपीयू क्लॉक की जांच करें और कर्नेल लोड करने और I/O को शुरू करने के लिए, पैरलललाइज़ेशन (मल्टी-कोर सपोर्ट की ज़रूरत होती है) पर विचार करें.
I/O की क्षमता को ऑप्टिमाइज़ करना
डिवाइस को तेज़ी से चालू करने के लिए, I/O की परफ़ॉर्मेंस को बेहतर बनाना ज़रूरी है. साथ ही, डिवाइस चालू होने के बाद ही ज़रूरी चीज़ें पढ़ी जानी चाहिए. Google Pixel पर, डिवाइस चालू होने के दौरान करीब 1.2 जीबी डेटा पढ़ा जाता है.
फ़ाइल सिस्टम को ट्यून करना
Linux kernel read ahead, फ़ाइल को शुरू से पढ़ने या ब्लॉक को क्रम से पढ़ने पर चालू होता है. इससे, खास तौर पर बूट करने के लिए I/O शेड्यूलर पैरामीटर को ट्यून करना ज़रूरी हो जाता है. बूट करने के लिए, सामान्य ऐप्लिकेशन के मुकाबले अलग तरह का वर्कलोड होता है.
बिना किसी रुकावट के (A/B) अपडेट करने की सुविधा वाले डिवाइसों को, पहली बार बूट करने पर फ़ाइल सिस्टम को ट्यून करने से काफ़ी फ़ायदा मिलता है. उदाहरण के लिए, Google Pixel पर 20 सेकंड. उदाहरण के लिए, हमने Google Pixel के लिए इन पैरामीटर को ट्यून किया है:
on late-fs
# boot time fs tune
# boot time fs tune
write /sys/block/sda/queue/iostats 0
write /sys/block/sda/queue/scheduler cfq
write /sys/block/sda/queue/iosched/slice_idle 0
write /sys/block/sda/queue/read_ahead_kb 2048
write /sys/block/sda/queue/nr_requests 256
write /sys/block/dm-0/queue/read_ahead_kb 2048
write /sys/block/dm-1/queue/read_ahead_kb 2048
on property:sys.boot_completed=1
# end boot time fs tune
write /sys/block/sda/queue/read_ahead_kb 512
...अन्य चीज़ें
- kernel config का इस्तेमाल करके, dm-verity हैश प्रीफ़ेच साइज़ को चालू करें DM_VERITY_HASH_PREFETCH_MIN_SIZE (डिफ़ॉल्ट साइज़ 128 है).
- फ़ाइल सिस्टम को बेहतर बनाने और हर बार बूट होने पर होने वाली ज़बरदस्ती की जांच को हटाने के लिए, नए ext4 जनरेशन टूल का इस्तेमाल करें. इसके लिए, BoardConfig.mk में TARGET_USES_MKE2FS सेट करें.
I/O का विश्लेषण करना
बूट के दौरान I/O गतिविधियों को समझने के लिए, kernel ftrace डेटा का इस्तेमाल करें. इसका इस्तेमाल systrace भी करता है:
trace_event=block,ext4 in BOARD_KERNEL_CMDLINE
हर फ़ाइल के लिए फ़ाइल ऐक्सेस को अलग-अलग करने के लिए, कर्नेल में ये बदलाव करें (सिर्फ़ डेवलपमेंट कर्नेल के लिए; प्रोडक्शन कर्नेल में इस्तेमाल न करें):
diff --git a/fs/open.c b/fs/open.c index 1651f35..a808093 100644 --- a/fs/open.c +++ b/fs/open.c @@ -981,6 +981,25 @@ } EXPORT_SYMBOL(file_open_root); +static void _trace_do_sys_open(struct file *filp, const struct open_how *how, long fd) +{ + char *buf; + char *fname; + + buf = kzalloc(PAGE_SIZE, GFP_KERNEL); + if (!buf) + return; + fname = d_path(&filp->f_path, buf, PAGE_SIZE); + + if (IS_ERR(fname)) + goto out; + + trace_printk("%s: open(\"%s\", %d, %d) fd = %ld, inode = %ld\n", + current->comm, fname, how->flags, how->mode, fd, filp->f_inode->i_ino); +out: + kfree(buf); +} + long do_sys_open(int dfd, const char __user *filename, int flags, umode_t mode) { struct open_flags op; @@ -1003,6 +1022,7 @@ } else { fsnotify_open(f); fd_install(fd, f); + _trace_do_sys_open(f, flags, mode, fd);
बूट की परफ़ॉर्मेंस का विश्लेषण करने के लिए, यहां दी गई स्क्रिप्ट का इस्तेमाल करें.
system/extras/boottime_tools/bootanalyze/bootanalyze.pyयह बूट होने में लगने वाले समय को मेज़र करता है. इसके लिए, बूट होने की प्रोसेस के अहम चरणों को अलग-अलग दिखाया जाता है.system/extras/boottime_tools/io_analysis/check_file_read.py boot_traceहर फ़ाइल के लिए ऐक्सेस की जानकारी देता है.system/extras/boottime_tools/io_analysis/check_io_trace_all.py boot_trace, सिस्टम-लेवल पर ब्रेकडाउन देता है.
init.*.rc को ऑप्टिमाइज़ करना
Init, फ़्रेमवर्क के बनने तक, कर्नेल से जुड़ा ब्रिज होता है. आम तौर पर, डिवाइसों को init के अलग-अलग चरणों में कुछ सेकंड लगते हैं.
एक साथ कई टास्क चलाना
Android के मौजूदा init प्रोसेस में, एक ही थ्रेड का इस्तेमाल किया जाता है. हालांकि, अब भी कुछ टास्क एक साथ किए जा सकते हैं.
- शेल स्क्रिप्ट सेवा में धीमे निर्देशों को लागू करें और बाद में किसी खास प्रॉपर्टी के लिए इंतज़ार करके उसमें शामिल हों. Android 8.0, इस्तेमाल के इस उदाहरण के लिए, एक नए
wait_for_propertyकमांड के साथ काम करता है. - init में धीमी कार्रवाइयों की पहचान करना. सिस्टम, init command/wait_for_prop या ऐसी किसी भी कार्रवाई को लॉग करता है जिसमें ज़्यादा समय लगता है. Android 8.0 में, 50 मि॰से॰ से ज़्यादा समय लेने वाली कोई भी कार्रवाई लॉग की जाती है. उदाहरण के लिए:
init: Command 'wait_for_coldboot_done' action=wait_for_coldboot_done returned 0 took 585.012ms
इस लॉग की समीक्षा करने से, आपको सुधार करने के अवसर मिल सकते हैं.
- सेवाएं शुरू करें और क्रिटिकल पाथ में, पेरिफ़रल डिवाइसों को जल्दी चालू करें. उदाहरण के लिए, कुछ SOC को SurfaceFlinger शुरू करने से पहले, सुरक्षा से जुड़ी सेवाएं शुरू करनी पड़ती हैं. जब ServiceManager "wait for service" दिखाता है, तो सिस्टम लॉग की समीक्षा करें — आम तौर पर, यह इस बात का संकेत होता है कि किसी दूसरी सेवा को पहले शुरू करना ज़रूरी है.
- init.*.rc में, इस्तेमाल नहीं की जा रही सेवाओं और निर्देशों को हटाएं. शुरुआती चरण में शुरू नहीं की गई किसी भी चीज़ को, बूट पूरा होने के बाद शुरू किया जाना चाहिए.
ध्यान दें: प्रॉपर्टी सेवा, init प्रोसेस का हिस्सा है. इसलिए, अगर init, पहले से मौजूद निर्देशों में व्यस्त है, तो बूट के दौरान setproperty को कॉल करने पर, देरी हो सकती है.
शेड्यूलर ट्यूनिंग का इस्तेमाल करना
जल्दी बूट करने के लिए, शेड्यूलर ट्यूनिंग का इस्तेमाल करें. Google Pixel फ़ोन का उदाहरण:
on init
# boottime stune
write /dev/stune/schedtune.prefer_idle 1
write /dev/stune/schedtune.boost 100
on property:sys.boot_completed=1
# reset stune
write /dev/stune/schedtune.prefer_idle 0
write /dev/stune/schedtune.boost 0
# or just disable EAS during boot
on init
write /sys/kernel/debug/sched_features NO_ENERGY_AWARE
on property:sys.boot_completed=1
write /sys/kernel/debug/sched_features ENERGY_AWAREकुछ सेवाओं को बूट करने के दौरान प्राथमिकता देने की ज़रूरत पड़ सकती है. उदाहरण:
init.zygote64.rc:
service zygote /system/bin/app_process64 -Xzygote /system/bin --zygote --start-system-server
class main
priority -20
user root
...ज़ाइगोट को जल्दी शुरू करना
फ़ाइल-आधारित एन्क्रिप्शन वाले डिवाइसों में, zygote-start ट्रिगर होने पर zygote को पहले से शुरू किया जा सकता है. डिफ़ॉल्ट रूप से, zygote को class main पर लॉन्च किया जाता है, जो zygote-start के बाद होता है. ऐसा करते समय, पक्का करें कि zygote को सभी सीपीयू में चलने की अनुमति दी गई हो. ऐसा इसलिए, क्योंकि गलत cpuset सेटिंग से zygote को कुछ खास सीपीयू में चलने के लिए मजबूर किया जा सकता है.
बैटरी सेवर मोड बंद करना
डिवाइस के बूट होने के दौरान, यूएफ़एस और/या सीपीयू गवर्नर जैसे कॉम्पोनेंट के लिए, बैटरी बचाने की सेटिंग बंद की जा सकती है.
चेतावनी: बेहतर परफ़ॉर्मेंस के लिए, चार्जर मोड में बैटरी बचाने की सुविधा चालू होनी चाहिए.
on init # Disable UFS powersaving write /sys/devices/soc/${ro.boot.bootdevice}/clkscale_enable 0 write /sys/devices/soc/${ro.boot.bootdevice}/clkgate_enable 0 write /sys/devices/soc/${ro.boot.bootdevice}/hibern8_on_idle_enable 0 write /sys/module/lpm_levels/parameters/sleep_disabled Y on property:sys.boot_completed=1 # Enable UFS powersaving write /sys/devices/soc/${ro.boot.bootdevice}/clkscale_enable 1 write /sys/devices/soc/${ro.boot.bootdevice}/clkgate_enable 1 write /sys/devices/soc/${ro.boot.bootdevice}/hibern8_on_idle_enable 1 write /sys/module/lpm_levels/parameters/sleep_disabled N on charger # Enable UFS powersaving write /sys/devices/soc/${ro.boot.bootdevice}/clkscale_enable 1 write /sys/devices/soc/${ro.boot.bootdevice}/clkgate_enable 1 write /sys/devices/soc/${ro.boot.bootdevice}/hibern8_on_idle_enable 1 write /sys/class/typec/port0/port_type sink write /sys/module/lpm_levels/parameters/sleep_disabled N
ज़रूरी न होने वाले इनिशलाइज़ेशन को टालना
ZRAM जैसे ग़ैर-ज़रूरी शुरू करने की प्रोसेस को boot_complete तक के लिए टाला जा सकता है.
on property:sys.boot_completed=1 # Enable ZRAM on boot_complete swapon_all /vendor/etc/fstab.${ro.hardware}
बूट ऐनिमेशन को ऑप्टिमाइज़ करना
बूट ऐनिमेशन को ऑप्टिमाइज़ करने के लिए, नीचे दिए गए सुझावों का इस्तेमाल करें.
रिकॉर्डिंग जल्दी शुरू करने की सुविधा कॉन्फ़िगर करना
Android 8.0 में, उपयोगकर्ता डेटा के partition को माउंट करने से पहले, बूट ऐनिमेशन को शुरू करने की सुविधा उपलब्ध है. हालांकि, Android 8.0 में नए ext4 टूल चेन का इस्तेमाल करने पर भी, सुरक्षा की वजहों से fsck समय-समय पर ट्रिगर होता रहता है. इस वजह से, बूटऐनिमेशन सेवा शुरू होने में देरी होती है.
बूटऐनिमेशन को जल्दी शुरू करने के लिए, fstab माउंट को दो चरणों में बांटें:
- शुरुआती चरण में, सिर्फ़ उन पार्टीशन (जैसे कि
system/औरvendor/) को माउंट करें जिनके लिए रन करने से पहले जांच करने की ज़रूरत नहीं होती. इसके बाद, बूट ऐनिमेशन सेवाओं और उनकी डिपेंडेंसी (जैसे कि servicemanager और surfaceflinger) को शुरू करें. - दूसरे चरण में, ऐसे पार्टिशन (जैसे कि
data/) माउंट करें जिनके लिए जांच की ज़रूरत है.
fsck के बावजूद, बूट ऐनिमेशन बहुत तेज़ी से (और एक ही समय में) शुरू हो जाएगा.
फ़िनिश क्लीन
डिवाइस बंद करने का सिग्नल मिलने के बाद, बूटऐनिमेशन का आखिरी हिस्सा चलता है. इस हिस्से की अवधि ज़्यादा होने पर, डिवाइस को चालू होने में ज़्यादा समय लग सकता है. तेज़ी से बूट होने वाले सिस्टम के लिए, लंबी अवधि वाले ऐनिमेशन की ज़रूरत नहीं होती. इनसे, सिस्टम में किए गए सुधारों को छिपाया जा सकता है. हमारा सुझाव है कि आप बार-बार चलने वाले लूप और फ़ाइनल, दोनों को छोटा बनाएं.
SELinux को ऑप्टिमाइज़ करना
बेहतर बूट समय के लिए, SELinux को ऑप्टिमाइज़ करने के लिए इन सुझावों का इस्तेमाल करें.
- क्लीन रेगुलर एक्सप्रेशन (रेगुलर एक्सप्रेशन) का इस्तेमाल करें.
file_contextsमेंsys/devicesके लिए, SELinux नीति से मैच करते समय, गलत तरीके से बनाए गए रेगुलर एक्सप्रेशन की वजह से काफ़ी ओवरहेड हो सकता है. उदाहरण के लिए, रेगुलर एक्सप्रेशन/sys/devices/.*abc.*(/.*)?, "abc" वाली सभी/sys/devicesसबडायरेक्ट्री को गलती से स्कैन करता है. इससे/sys/devices/abcऔर/sys/devices/xyz/abc, दोनों के लिए मैच की सुविधा चालू हो जाती है. इस रेगुलर एक्सप्रेशन को/sys/devices/[^/]*abc[^/]*(/.*)?में बदलने पर, सिर्फ़/sys/devices/abcके लिए मैच की सुविधा चालू हो जाएगी. - लेबल को genfscon में ले जाएं. SELinux की मौजूदा सुविधा, फ़ाइल से मैच करने वाले प्रीफ़िक्स को SELinux बाइनरी में, कर्नेल में भेजती है. इसके बाद, कर्नेल उन्हें कर्नेल से जनरेट किए गए फ़ाइल सिस्टम पर लागू करता है. इससे, गलत लेबल वाली कर्नेल से बनाई गई फ़ाइलों को ठीक करने में भी मदद मिलती है. साथ ही, रीलेबल करने से पहले, उपयोगकर्ता स्पेस की प्रोसेस के बीच इन फ़ाइलों को ऐक्सेस करने की कोशिश करने पर, रेस कंडीशन से भी बचा जा सकता है.
टूल और तरीके
ऑप्टिमाइज़ेशन टारगेट के लिए डेटा इकट्ठा करने में मदद पाने के लिए, नीचे दिए गए टूल का इस्तेमाल करें.
Bootchart
Bootchart, पूरे सिस्टम के लिए सभी प्रोसेस के सीपीयू और I/O लोड का ब्रेकडाउन उपलब्ध कराता है. इसके लिए, सिस्टम इमेज को फिर से बनाने की ज़रूरत नहीं होती. साथ ही, systrace का इस्तेमाल करने से पहले, इसकी मदद से तुरंत जांच की जा सकती है.
बूटचार्ट चालू करने के लिए:
adb shell 'touch /data/bootchart/enabled'adb reboot
डिवाइस के चालू होने के बाद, बूट चार्ट फ़ेच करें:
$ANDROID_BUILD_TOP/system/core/init/grab-bootchart.sh
डेटा इकट्ठा होने की प्रोसेस पूरी होने के बाद, /data/bootchart/enabled को मिटाएं, ताकि हर बार डेटा इकट्ठा न हो.
bootchart.png मौजूद नहीं है, तो ये काम करें:
- ये कमांड चलाएं:
sudo apt install python-is-python3cd ~/Documentsgit clone https://github.com/xrmx/bootchart.gitcd bootchart/pybootchartguimv main.py.in main.py -
$ANDROID_BUILD_TOP/system/core/init/grab-bootchart.shको अपडेट करेंpybootchartguiकी लोकल कॉपी पर ले जाने के लिए (~/Documents/bootchart/pybootchartgui.pyपर मौजूद)
Systrace
Systrace की मदद से, बूट अप के दौरान कर्नेल और Android, दोनों के ट्रैक इकट्ठा किए जा सकते हैं. systrace के विज़ुअलाइज़ेशन से, बूट-अप के दौरान किसी खास समस्या का विश्लेषण करने में मदद मिल सकती है. (हालांकि, पूरे बूट के दौरान औसत संख्या या इकट्ठा की गई संख्या देखने के लिए, सीधे कर्नेल ट्रेस को देखना आसान है).
बूट-अप के दौरान systrace चालू करने के लिए:
-
frameworks/native/cmds/atrace/atrace.rcमें, बदलें:write /sys/kernel/debug/tracing/tracing_on 0 write /sys/kernel/tracing/tracing_on 0
इन कार्रवाइयों के लिए नीचे दिया गया तरीका अपनाएं:
# write /sys/kernel/debug/tracing/tracing_on 0 # write /sys/kernel/tracing/tracing_on 0
device.mkफ़ाइल में, यह लाइन जोड़ें:PRODUCT_PROPERTY_OVERRIDES += debug.atrace.tags.enableflags=802922 PRODUCT_PROPERTY_OVERRIDES += persist.traced.enable=0
- डिवाइस की
BoardConfig.mkफ़ाइल में, ये चीज़ें जोड़ें:BOARD_KERNEL_CMDLINE := ... trace_buf_size=64M trace_event=sched_wakeup,sched_switch,sched_blocked_reason,sched_cpu_hotplug
- डिवाइस के हिसाब से बनाई गई
init.rcफ़ाइल में, ये चीज़ें जोड़ें:on property:sys.boot_completed=1 // This stops tracing on boot complete write /d/tracing/tracing_on 0 write /d/tracing/events/ext4/enable 0 write /d/tracing/events/f2fs/enable 0 write /d/tracing/events/block/enable 0
-
बूट अप होने के बाद, ट्रेस फ़ेच करें:
adb root && adb shell atrace --async_stop -z -c -o /data/local/tmp/boot_traceadb pull /data/local/tmp/boot_trace$ANDROID_BUILD_TOP/external/chromium-trace/systrace.py --from-file=boot_trace
इससे ट्रैकिंग की सुविधा चालू हो जाती है. यह सुविधा डिफ़ॉल्ट रूप से बंद रहती है.
I/O के बारे में ज़्यादा जानकारी के लिए, ब्लॉक और ext4 और f2fs भी जोड़ें.