Flytouch 2/Superpad III и попытка сэкономить байты в ядре Линукс

от автора

Я планировал написать данную статью уже давно, но в последние месяцы
никак не мог выкроить достаточно времени. Пока я размышлял над статьёй,
делал примеры и проверял свои догадки, на Хабре уже обсудили
константность — [1][2].

Ради забавы попробуем проделать подобную экономию не со сферическим
проектом в вакууме, а с самым что ни на есть живым и грандиозным
проектом — с ядром Линукс!

Статья имеет следующую структуру:

  1. Предыстория
  2. Планшет Flytouch 2/Superpad III
  3. Константность
  4. Измерения
  5. Заключение
  6. Список источников

По идее статью можно было бы разбить на две части: технические детали про планшет и использование extern const в ядре Линукс. Мне кажется, что тогда полноценных публикаций не получилось бы, поэтому я объединил весь материал в одну статью.

Предыстория ^

В далёком 2011 году я начитался на хабре обзоров про китайпады — [3][4][5]. Естественно мне захотелось такого чуда и я его заказал.

Немножко поигрался, поглядел кино, поразгадывал маджонг и… И вскоре моё мнение стало альтернативным предыдущему — [6].

После этого было решено изучать планшет с точки зрения программиста, а именно — установить туда обычный Линукс, что-нибудь покомпилировать.

Планшет Flytouch 2/Superpad III ^

Всё описанное справедливо для китайпада, у которого присутствуют такие опознавательные знаки:

показать

DF-MID10-IX210-V1.1
2010-01-20

XW11070501B512M03101

Здесь [8] доступно гик-порно. На фотках отчётливо можно разглядеть выводы JTAG & USART. Также обратите внимание на маркировку.

Прошивка ^

Для прошивки устройства надо на СД-карточку с файловой системой FAT загрузить файлы с именами firmware2, firmware-discovery, bootloader-discovery; вставить её в разьём и включить планшет. (У меня заработало с флешкой 256МБ и файловой системой W95 FAT32 — идентификатор 0xB в fdisk).

Понадобится рабочая прошивка. Хотя предмет статьи давно устарел, ещё можно найти рабочий вариант здесь — [12]. Я взял версию Axlien.

Предназначение файлов:

  1. файлы *-discovery — это просто zip-архивы с нужными данными для обновления. firmware-discovery содержит всё, что касается Андроида. Про bootloader ничего сказать не могу — он мне не попадался, но по названию можно догадываться, что это обновление загрузчика — не ясно только, какого именно;
  2. файл firmware2 представляет наибольший интерес. На тот момент нигде в сети я не нашёл упоминания формата этого файла. Я подумал, ну не могут же китайцы, да ещё и для такого дешёвого барахла, создать мощную систему на основе криптографии… и оказался прав! Первые 192 байта — это просто мусор. Можно взять firmware2 от рабочей прошивки и затереть указанный заголовок нулями — процесс обновления прошивки всё равно запуститься [10]. Далее идёт обычный U-Boot образ, параметры которого можно получить с помощью mkimage.

Таким образом, если собрать своё собственное рабочее ядро (zImage) для планшета, создать U-Boot образ с теми же параметрами и добавить 192-байтный заголовок, то удастся загрузить свою версию Линукса.

Также имеется возможность поиграть с «официальным» firmware2. Имеются инструменты для распаковки и обратной запаковки файла zImage [9]. Можно распаковать и изучить сценарии обновления устройства, в частности, понять, что обновляет bootloader-discovery и как. С другой стороны, есть возможность отключить упомянутые сценарии и оказаться в оболочке busybox.

Сборка ядра ^

В поисках исходников ядра под данный процессор я наткнулся на обсуждение [13], где некто atp_uestc сообщил об открытии исходных кодов ядра для планшета ZT-180.

Сейчас мне уже сложно вспомнить хронологию событий и какое же ядро я пытался грузить сперва: то ли «ванильное» отсюда — [14], то ли модифицированное отсюда — [15]. Запомнился только результат — не заработало, процесс загрузки замирал на «Decompressing kernel…»

Спустя некоторое время в обсуждении появился yuray, который добавлял поддержку данного процессора в третью версию ядра линукс. Именно эту версию ядра мне и удалось запустить на китайпаде.

Скачать исходные тексты ядра версии 3.4.х с kernel.org. На момент написания статьи успешно собиралась версия 3.4.108 и ниже. Загрузить и наложить заплатки для ядра от yuray отсюда rtck.org/zt180/patches/zt180_b0_3.4.patch.xz:

$ cd /path/to/linux-3.4.x $ patch -p1 -i /path/to/zt180_b0_3.4.patch 

Настроить конфиг. Правильного конфига под этот китайпад я не встречал, поэтому предлагаю взять dot-config отсюда [19]:

$ make V=1 ARCH=arm CROSS_COMPILE=/path/to/toolchain/arm-infotm-linux-gnueabi-  oldconfig 

Для ядра потребуется initramfs. Я пользователь ОС Slackware GNU/Linux, поэтому решено было взять initrd установщика из неё, чтобы в последствии установить на планшет слаку. Загружаем uinitrd-kirkwood.img [16], там же в файле kernels/README.txt приведены команды для распаковки образа. Из оригинальной слаки образ не подойдёт, т.к. содержит бинарники под x86/amd64, но нам нужны под ARM.

Создаём папку, распаковываем в неё образ:

$ mkdir -p /path/to/uinitrd.extracted $ pushd !$ $ dd if=/path/to/uinitrd-kirkwood.img bs=64 skip=1 | gzip -dc | sudo cpio -div $ popd 

Указываем в конфиге путь к папке с образом:
#…
CONFIG_BLK_DEV_INITRD=y
CONFIG_INITRAMFS_SOURCE="/path/to/uinitrd.extracted"
CONFIG_INITRAMFS_ROOT_UID=0
CONFIG_INITRAMFS_ROOT_GID=0
#…

Компилируем ядро в два прохода:

# clean initrd directory $ sudo rm -rf /path/to/uinitrd.extracted/lib/{modules, firmware} # build kernel for the first time $ make KALLSYMS_EXTRA_PASS=1 ARCH=arm CROSS_COMPILE=/path/to/toolchain/arm-infotm-linux-gnueabi- -j2 zImage # build modules against the kernel $ make KALLSYMS_EXTRA_PASS=1 ARCH=arm CROSS_COMPILE=/path/to/toolchain/arm-infotm-linux-gnueabi- -j2 modules # install modules into initramfs dir $ sudo make ARCH=arm CROSS_COMPILE=/path/to/toolchain/arm-infotm-linux-gnueabi- -j2 modules_install INSTALL_MOD_PATH=/path/to/uinitrd.extracted/ # install firmware to initramfs folder $ sudo make ARCH=arm CROSS_COMPILE=/path/to/toolchain/arm-infotm-linux-gnueabi- -j2 firmware_install INSTALL_MOD_PATH=/path/to/uinitrd.extracted/ # remove unnecessary files $ sudo rm -rf /path/to/uinitrd.extracted/lib/modules/3.4.*/{build,source} # build the kernel again, with initramfs dir contains modules $ make KALLSYMS_EXTRA_PASS=1 ARCH=arm CROSS_COMPILE=/path/to/toolchain/arm-infotm-linux-gnueabi- -j2 zImage 

Может вылезти ошибка [17][18]. Мне помогло решение, которое и предлагается в тексте ошибки: параметр KALLSYMS_EXTRA_PASS=1.

Потом создаём firmware2:

# make u-boot image $ /path/to/mkimage -A arm -C none -O linux -T kernel -a 0x40008000 -e 0x40008000 -n linux-3.4 -d arch/arm/boot/zImage arch/arm/boot/uImage # make firmware2 $ cat /path/to/firmware2.header arch/arm/boot/uImage > firmware2 

и загружаём с неё планшет.

Для автоматизации сборки был написан сценарий [19]. Использовалась следующая команда:

$ ( time ./my_build.sh ) |& tee `date +%M%H_%F`-build-log.txt 

Загрузил — не работает ЮСБ, точнее, на портах отсутствовало питание. Просил помощи у yuray, на что он ответил, что скорее всего схема моего китайпада имеет другую разводку, отличную от ZT-180.

Потом нашёл эту великолепную статью [11] о том, как дизассемблировать части рабочего ядра. Решил восстанавливать сишный код функций «оригинальной» прошивки, имеющих отношение к USB, и сравнивать их с версиями в исходном коде. К моему разочарованию, функции совпадали с точностью до диграфов. Однако на рабочем ядре имелись функции, которых не было в исходных текстах. Видимо там кроется некоторая магия, которая вселяет жизнь в USB-порты, но её я так и оставил под завесой тайны…

В процессе изучения восстановленного кода ассемблера, я подметил один интересный момент: во многих функциях после их эпилога шли странные инструкции:

/* ... */ c039e32c:	e59f0030 	ldr	r0, [pc, #48]	; c039e364 <_binary_0xc039e2e8_imapx200_decode_suspend_start+0x7c> /* ... */ c039e360:	e89da800 	ldm	sp, {fp, sp, pc} c039e364:	f0200000 	undefined instruction 0xf0200000 c039e368:	c24b1c9c 	subgt	r1, fp, #39936	; 0x9c00 

Видите 0xF0200000? Странная неопределённая инструкция… да ещё и встречается в нескольких местах. Это некий базовый адрес, подумал я, а значит, что теоретически можно с каждой такой функции сэкономить 4 (четыре!) байта: разместить это значение в одном месте, а все функции будут грузить его по одному адресу — завести константу!

Константность ^

У Мейерса [20] очень хорошо всё расписано. Приведу лишь некоторые пояснения.

Компиляторы языков C и C++ однопроходные, а значит не могут самостоятельно оптимизировать константы. Как минимум требуется помощь со стороны компоновщика, например, параметр /Gw [1].

Эта же однопроходная слабость может сыграть на руку: использовать extern const [1]. Идея заключается в том, чтобы в заголовочном файле описать постоянную следующим образом:

#ifndef MY_CONST_H #define MY_CONST_H  extern const int my_constant;  #endif /* MY_CONST_H */ 

Затем в некотором файле написать строчку:

const int my_constant = 42; 

это можно разместить в любой единице трансляции. Теперь компилятор не сможет зашить значение постоянной в код, но вынужден будет сделать на неё некоторую ссылку. Компоновщик же получит только один объектный файл, в котором найдётся значение постоянной, после чего подставит конечные адреса в код. Конечно могут быть варианты, но в общем случае компилятор в одиночку беспомощен перед таким трюком и вынужден поступать так, как описано выше. Однако есть маленький нюанс [21].

Для проведения эксперимента потребуется внести следующие правки [19]:

показать

diff -ru ./linux-3.4.108/arch/arm/mach-imapx200/Makefile ../linux-3.4.108/arch/arm/mach-imapx200/Makefile --- ./linux-3.4.108/arch/arm/mach-imapx200/Makefile	2015-09-09 12:17:52.483020878 +0300 +++ ../linux-3.4.108/arch/arm/mach-imapx200/Makefile	2015-09-08 17:11:45.775035963 +0300 @@ -6,7 +6,7 @@    #IMAPX200 support files   -obj-$(CONFIG_CPU_IMAPX200)		+= irq.o clock.o time.o devices.o pwm.o +obj-$(CONFIG_CPU_IMAPX200)		+= irq.o clock.o time.o devices.o pwm.o constants.o       diff -ru ./linux-3.4.108/arch/arm/mach-imapx200/include/mach/imapx_base_reg.h ../linux-3.4.108/arch/arm/mach-imapx200/include/mach/imapx_base_reg.h --- ./linux-3.4.108/arch/arm/mach-imapx200/include/mach/imapx_base_reg.h	2015-09-09 12:17:52.498020874 +0300 +++ ../linux-3.4.108/arch/arm/mach-imapx200/include/mach/imapx_base_reg.h	2015-09-08 17:11:45.778035706 +0300 @@ -15,13 +15,23 @@  #define IMAPX200_SDRAM_PA     		(0x40000000)    /************************Virtual address for peripheral*************************/ -#define IMAP_VA_SYSMGR       			IMAP_ADDR(0x00200000) -#define IMAP_VA_IRQ            	 		IMAP_ADDR(0x00000000) -#define IMAP_VA_TIMER        			IMAP_ADDR(0x00300000) -#define IMAP_VA_WATCHDOG        		IMAP_ADDR(0x00600000) -#define IMAP_VA_GPIO         			IMAP_ADDR(0x00400000) -#define IMAP_VA_NAND				IMAP_ADDR(0x00500000) -#define IMAP_VA_FB				IMAP_ADDR(0x00700000) +#if defined(IMAP_USE_MACRO_CONSTANTS) || defined(__ASSEMBLY__)  +#   define IMAP_VA_SYSMGR       			IMAP_ADDR(0x00200000) +#   define IMAP_VA_IRQ            	 		IMAP_ADDR(0x00000000) +#   define IMAP_VA_TIMER        			IMAP_ADDR(0x00300000) +#   define IMAP_VA_WATCHDOG        		IMAP_ADDR(0x00600000) +#   define IMAP_VA_GPIO         			IMAP_ADDR(0x00400000) +#   define IMAP_VA_NAND				IMAP_ADDR(0x00500000) +#   define IMAP_VA_FB				IMAP_ADDR(0x00700000) +#else +extern const void __iomem __force * const IMAP_VA_SYSMGR; +extern const void __iomem __force * const IMAP_VA_IRQ; +extern const void __iomem __force * const IMAP_VA_TIMER; +extern const void __iomem __force * const IMAP_VA_WATCHDOG; +extern const void __iomem __force * const IMAP_VA_GPIO; +extern const void __iomem __force * const IMAP_VA_NAND; +extern const void __iomem __force * const IMAP_VA_FB; +#endif /* defined(IMAP_USE_MACRO_CONSTANTS) || defined(__ASSEMBLY__) */      #define PERIPHERAL_BASE_ADDR_PA			(0x20C00000) diff -ru ./linux-3.4.108/arch/arm/plat-imap/cpu.c ../linux-3.4.108/arch/arm/plat-imap/cpu.c --- ./linux-3.4.108/arch/arm/plat-imap/cpu.c	2015-09-09 12:17:52.607021038 +0300 +++ ../linux-3.4.108/arch/arm/plat-imap/cpu.c	2015-09-08 17:18:30.646035384 +0300 @@ -1,3 +1,5 @@ +#define IMAP_USE_MACRO_CONSTANTS +  /********************************************************************************   ** linux-2.6.28.5/arch/arm/plat-imap/cpu.c   ** diff -ru ./linux-3.4.108/arch/arm/plat-imap/gpio.c ../linux-3.4.108/arch/arm/plat-imap/gpio.c --- ./linux-3.4.108/arch/arm/plat-imap/gpio.c	2015-09-09 12:17:52.614020935 +0300 +++ ../linux-3.4.108/arch/arm/plat-imap/gpio.c	2015-09-08 17:18:38.566035206 +0300 @@ -1,3 +1,5 @@ +#define IMAP_USE_MACRO_CONSTANTS +  /* arch/arm/plat-imapx200/gpiolib.c   *   * Copyright 2008 Openmoko, Inc. diff -ru ./linux-3.4.108/arch/arm/plat-imap/pm_imapx200.c ../linux-3.4.108/arch/arm/plat-imap/pm_imapx200.c --- ./linux-3.4.108/arch/arm/plat-imap/pm_imapx200.c	2015-09-09 12:17:52.631020886 +0300 +++ ../linux-3.4.108/arch/arm/plat-imap/pm_imapx200.c	2015-09-08 17:18:52.102036874 +0300 @@ -1,3 +1,5 @@ +#define IMAP_USE_MACRO_CONSTANTS +  #include <linux/init.h>  #include <linux/suspend.h>  #include <linux/serial_core.h> diff -ru ./linux-3.4.108/drivers/video/infotm/imapfb.c ../linux-3.4.108/drivers/video/infotm/imapfb.c --- ./linux-3.4.108/drivers/video/infotm/imapfb.c	2015-09-09 12:17:53.350020920 +0300 +++ ../linux-3.4.108/drivers/video/infotm/imapfb.c	2015-09-08 17:34:34.814042727 +0300 @@ -1,3 +1,5 @@ +#define IMAP_USE_MACRO_CONSTANTS +  /*****************************************************************************   ** drivers/video/infotm/imapfb.c  **  diff -ru --new-file ./linux-3.4.108/arch/arm/mach-imapx200/constants.c ../linux-3.4.108/arch/arm/mach-imapx200/constants.c --- ./linux-3.4.108/arch/arm/mach-imapx200/constants.c	1970-01-01 03:00:00.000000000 +0300 +++ ../linux-3.4.108/arch/arm/mach-imapx200/constants.c	2015-09-09 14:56:53.513487879 +0300 @@ -0,0 +1,11 @@ +#include <linux/compiler.h> + +#include <mach/imapx_base_reg.h> + +const void __iomem __force * const IMAP_VA_SYSMGR = IMAP_ADDR(0x00200000); +const void __iomem __force * const IMAP_VA_IRQ = IMAP_ADDR(0x00000000); +const void __iomem __force * const IMAP_VA_TIMER = IMAP_ADDR(0x00300000); +const void __iomem __force * const IMAP_VA_WATCHDOG = IMAP_ADDR(0x00600000); +const void __iomem __force * const IMAP_VA_GPIO = IMAP_ADDR(0x00400000); +const void __iomem __force * const IMAP_VA_NAND = IMAP_ADDR(0x00500000); +const void __iomem __force * const IMAP_VA_FB = IMAP_ADDR(0x00700000); 

Измерения ^

Было решено измерять два показателя:

  • размер файла ядра vmlinux;
  • подсчитывать размер функций непосредственно во время работы ОС.

Сценарий, который подсчитывает размер всех функций содержащих подслово imap, находится здесь [19].

Результаты следующие:

показать

# diff -ru 1.log 2.log
 --- 1.log       2015-09-11 16:57:28.430158628 +0300    +++ 2.log       2015-09-11 16:57:20.627161803 +0300    @@ -1,4 +1,4 @@ -Data Size:    17844272 Bytes = 17426.05 kB = 17.02 MB +Data Size:    17843856 Bytes = 17425.64 kB = 17.02 MB  Load Address: 0x40008000  Entry Point:  0x40008000  --- 22012015-rom/sizes.txt	2015-01-28 15:25:51.107945315 +0300 +++ 28012015-rom/sizes.txt	2015-01-28 15:15:48.052949785 +0300 @@ -1,25 +1,25 @@ -imapx200_timer_mask - 44 -imapx200_timer_unmask - 52 -imapx200_timer_ack - 44 +imapx200_timer_mask - 52 +imapx200_timer_unmask - 60 +imapx200_timer_ack - 52  imapx200_irq_add - 24  imapx200_irq_init - 32  imapx200_irq_wake - 44 -imapx200_irq_unmask - 136 -imapx200_irq_mask - 132 -imapx200_irq_ack - 120 +imapx200_irq_unmask - 172 +imapx200_irq_mask - 160 +imapx200_irq_ack - 152  imap_clk_enable - 60  imap_clkcon_enable - 76 -imapx200_gettimeoffset - 64 -imapx200_timer_setup - 164 -imapx200_timer_interrupt - 64 +imapx200_gettimeoffset - 68 +imapx200_timer_setup - 168 +imapx200_timer_interrupt - 76  imap_pwm_suspend - 188  imap_pwm_resume - 184  imap_pwm_start - 164  imap_timer_setup - 404  imap_default_idle - 20 -imapx_poweroff - 56 -imapx_reset - 64 -imapx200_idle - 52 +imapx_poweroff - 88 +imapx_reset - 68 +imapx200_idle - 48  imap_set_board - 112  imapx200_gpio_setpull_updown - 56  imapx200_gpio_getpull_updown - 36 @@ -48,7 +48,7 @@  imapx200_pm_prepare - 24  imapx200_pm_finish - 20  imapx200_pm_do_save - 88 -imapx200_pm_enter - 436 +imapx200_pm_enter - 500  imapx200_pm_configure_extint - 20  imapx200_pm_prepare - 24  imapx200_pm_init - 80 @@ -86,8 +86,8 @@  imapfb_resume - 196  imapfb_suspend - 204  imapfb_backlight_power_supply - 20 -imapfb_set_clk - 44 -imapfb_set_gpio - 88 +imapfb_set_clk - 52 +imapfb_set_gpio - 96  imapfb_set_brightness - 36  imapfb_lcd_power_supply - 32  con_get_unimap - 360 @@ -153,7 +153,7 @@  imap_nand_irq - 92  ehci_imapx200_drv_remove - 80  ehci_imapx200_init - 1148 -ehci_imapx200_drv_probe - 552 +ehci_imapx200_drv_probe - 556  ohci_hcd_imapx200_drv_remove - 80  ohci_imapx200_start - 100  ohci_hcd_imapx200_drv_probe - 512 @@ -196,22 +196,22 @@  imapx200_i2c_probe - 824  imapx200_i2c_irq - 920  imapx200_decode_poll - 184 -imapx200_decode_suspend - 132 +imapx200_decode_suspend - 44  imapx200_decode_resume - 68 -imapx200_decode_release - 192 +imapx200_decode_release - 104  imapx200_decode_open - 124  imapx200_decode_remove - 208  imapx200_decode_ioctl - 384 -imapx200_decode_probe - 908 +imapx200_decode_probe - 816  imapx200_decode_irq_handle - 264  imapx200_encode_poll - 84 -imapx200_encode_suspend - 116 +imapx200_encode_suspend - 28  imapx200_encode_ioctl - 348  imapx200_encode_resume - 68 -imapx200_encode_release - 188 +imapx200_encode_release - 100  imapx200_encode_open - 120  imapx200_encode_remove - 232 -imapx200_encode_probe - 1080 +imapx200_encode_probe - 988  imapx200_encode_irq_handle - 136  sdhci_imap_set_clk_src - 52  sdhci_imap_resume - 36 @@ -220,7 +220,7 @@  sdhci_imap_get_timeout_clk - 40  imapfb_probe - 2952  imapfb_init - 28 -sdhci_imap_probe - 604 +sdhci_imap_probe - 608  sdhci_imap_remove - 20  name_imapx200 - 12  imapfb_a1rgb232_8 - 48 @@ -342,8 +342,8 @@  __kstrtab_imap_get_reservemem_paddr - 26  __kstrtab_con_copy_unimap - 16  __kstrtab_con_set_default_unimap - 23 -imapx200_init_clocks - 1120 -imapx200_timer_init - 120 +imapx200_init_clocks - 1104 +imapx200_timer_init - 124  imapx200_register_device - 56  imap_init_pwm - 308  imapx200_fixup - 36 @@ -542,7 +542,7 @@  imapx200_i2c_driver - 116  imapx200_i2c_driver - 116  imapx200_decode_driver - 80 -imapx200_decode_fops - 144 +imapx200_decode_fops - 148  imapx200_encode_driver - 80  imapx200_encode_fops - 180  sdhci_imap_driver - 80 

Размер ядра в целом уменьшился на 416, но выигрыш оказался не таким, каким я его ожидал: некоторые функции прибавили в весе.

Возможно более опытный читатель знает причину, но мне на тот момент было не всё так ясно. Рассмотрим исходные тексты ассемблера функции imapx200_timer_ack, размер которой увеличился на 8 байт после модификации:

показать

--- 0xc0019c40-t-imapx200_timer_ack-2.listing	2015-11-18 22:12:24.196113878 +0300 +++ 0xc0019c50-imapx200_timer_ack-2.listing	2015-11-18 22:12:24.297113880 +0300 @@ -9,10 +9,12 @@  XXXXXXXX:	e92dd800 	push	{fp, ip, lr, pc}  XXXXXXXX:	e24cb004 	sub	fp, ip, #4	; 0x4  XXXXXXXX:	e1a00000 	nop			(mov r0,r0) +c0019c60:	e59f2018 	ldr	r2, [pc, #24]	; c0019c80 <_binary_0xc0019c50_imapx200_timer_ack_start+0x30>  XXXXXXXX:	e590Y000 	ldr	rY, [r0]  XXXXXXXX:	e3a0X001 	mov	rX, #1	; 0x1 -c0019c58:	e3a0120f 	mov	r1, #-268435456	; 0xf0000000 +c0019c6c:	e5921000 	ldr	r1, [r2]  XXXXXXXX:	e1a0XX1Y 	lsl	rX, rX, rY  XXXXXXXX:	e581X010 	str	rX, [r1, #16]  XXXXXXXX:	e581X000 	str	rX, [r1]  XXXXXXXX:	e89da800 	ldm	sp, {fp, sp, pc} +c0019c80:	c04e858c 	subgt	r8, lr, ip, lsl #11 

Первое, что бросается в глаза: сам базовый адрес настолько удачный, что запись оного в регистр умещалась в четыре байта команды mov:

c0019c58:	e3a0120f 	mov	r1, #-268435456	; 0xf0000000 

Второе: после модификации компилятору пришлось добавить за эпилогом функции адрес нашей новой постоянной, т.к. его значение менее удобное — четыре байта раз.

И последнее: считывать значения напрямую из памяти невозможно, поэтому сначала записывается адрес постоянной в регистр:

c0019c60:	e59f2018 	ldr	r2, [pc, #24]	; c0019c80 

— четыре байта два.

Заключение ^

Я думаю, что объективным выводом будет то, что нужно знать о последствиях определения постоянных как в виде макросов, так и с помощью const переменных.

Лично я мечтаю о том, чтобы компилятор (пусть и в связке с компоновщиком) самостоятельно принимал решение о встраивании значения постоянной без сложностей с extern const.

Спасибо за внимание!

Список источников ^

  1. 1 2 3 Вычислите длину окружности
  2. ^  И ещё раз про уникальные константы
  3. ^  История покупки и опыт использования планшетного ПК Zenithink ZT-180
  4. ^  Обзор планшета Zenithink Zt-180 10"
  5. ^  страница уже не доступна, ищите в различных архивах паутины по запросу habrahabr.ru/blogs/iTablet/110714
  6. ^  страница уже не доступна, ищите в различных архивах паутины по запросу www.good-review.ru/pandawill/2011/02/21/obzor-kitayskogo-plansheta-zenithink-zt-180-10.html
  7. Переселение души: linux на android планшете
  8. ^  Фото внутренностей
  9. ^  Zimage unpack and pack tools
  10. ^  Decompiling ‘firmware2’ and ‘firmware_discovery’
  11. ^  Disassembly: Smashing the Android Kernel for Fun and Overclock
  12. ^  Тема на forum.china-iphone.ru, посвящённая планшету
  13. ^  Open source project
  14. ^  github.com/atpboy444/ZT-180
  15. ^  github.com/dandel/linux-2.6.32.y
  16. ^  SlackwareARM-14.1
  17. ^  https://github.com/djwillis/meta-raspberrypi/issues/38
  18. ^  https://lkml.org/lkml/2012/7/6/260
  19. 1 2 3 4 github.com/gshep/flytouch2-helper-scripts
  20. ^  Скотт Мэйерс. Эффективное использование C++. 55 верных советов улучшить структуру и код ваших программ
  21. ^  Initialization order of globals

ссылка на оригинал статьи http://habrahabr.ru/post/270885/


Комментарии

Добавить комментарий

Ваш адрес email не будет опубликован. Обязательные поля помечены *