الخطأ: النجاح وما العمل حيال ذلك

صورة







يتم إخبار جميع المبرمجين المبتدئين دائمًا بأهمية الإبلاغ الصحيح عن الأخطاء. يقولون دائمًا أنه إذا لم يستطع البرنامج فعل شيء ما ، فيجب أن يخبرنا بوضوح ودون لبس عن سبب حدوثه. يتحدثون عن أهمية التحكم في القيمة المعادة للوظائف المسماة. علاوة على ذلك ، حتى المترجمون تعلموا إصدار تحذيرات إذا تم تجاهل القيمة التي تم إرجاعها بواسطة وظائف معينة. يجب أن أقول إن أهمية معالجة الأخطاء من قبل المبرمجين المعاصرين مقبولة. يؤدي هذا في بعض الأحيان إلى حوادث مسلية ، كما هو الحال في KDPV (مأخوذ هنا). في الحياة الواقعية ، كان علي أن أتعامل مع مثل هذه الرسائل التشخيصية الغريبة عدة مرات. أريد أن أخبركم عن الحالة الأخيرة وطرق التغلب على مثل هذا التشخيص. إذا كنت مهتمًا ، فأنت مرحب بك تحت القط. بالتأكيد لن يكتشف المبرمجون المتمرسون أي شيء جديد لأنفسهم ، لكنهم بالتأكيد سيكونون قادرين على التفلسف حول تطوير البرمجيات.







بالنسبة لأولئك الذين لديهم "الكثير من الحروف"
  • من المهم التعامل بشكل صحيح مع القيم التي يتم إرجاعها بواسطة read ()
  • ( OpenSource )
  • - . « »


بشكل عام ، لدي أخبار حزينة. لن يكون هناك المزيد من الصور. سننزل إلى مستوى وحدة التحكم في نظام Linux ونعيش هناك. في هذه الحالة سنبتهج. بالنسبة للمشروع الذي تعمل به ، فإن محمل U-Boot معروف جيدًا . مشروع مفتوح المصدر مدعوم من DENX Software Engineering... لذلك ، سنكون سعداء لأن لدينا وحدة تحكم وبيئة نظام ، وبشكل عام ، الحياة على قدم وساق. لأنه عند العمل مع هذا المشروع ، كقاعدة عامة ، لا يوجد شيء مثل هذا - مناطق الذاكرة المستمرة ، ونقل البايت من مكان إلى آخر ، وانتظار أن يكون المحيط جاهزًا. ولكن ، بالمناسبة ، هذا الجزء قد اكتمل بالفعل ويوجد محمل إقلاع صالح لقطعة الأجهزة. حان الوقت للبدء في استخدام الزخارف التي تسمح لمبرمجي التطبيقات بالتأثير بطريقة ما على عملية تمهيد النظام. لا شيء يبشر بالخير بالنسبة للمشاكل. تم حل المشكلة منذ فترة طويلة ويتم استخدامها بنشاط من قبل مشاريع شعبية مثل OpenWRT والعديد من المشاريع الأخرى ، الأقل شهرة.







. U-Boot . . fw_printenv fw_setenv Linux. . . « ». ? . «fw_printenv», - .







localhost ~ # fw_printenv
Cannot open /dev/mtd1: No such file or directory
localhost ~ # fw_printenv --help
Usage: fw_printenv [OPTIONS]... [VARIABLE]...
Print variables from U-Boot environment

 -h, --help           print this help.
 -v, --version        display version
 -c, --config         configuration file, default:/etc/fw_env.config
 -n, --noheader       do not repeat variable name in output
 -l, --lock           lock node, default:/var/lock
      
      





. . . « » , . /etc/fw_env.config. . , ( ) U-Boot , . uboot.env , vfat ( FAT-32). . U-Boot , . . Linux. c uboot.env, , , /boot. 11 12 (/dev/mtd1 /dev/mdt2 ) 30 (/boot/uboot.env) .







# VFAT example
/boot/uboot.env 0x0000          0x4000
      
      





. . .







localhost ~ # fw_printenv
Read error on /boot/uboot.env: Success
      
      





, . , Linux’ — . , « » — root’. . , ( ) ? ? U-Boot’ «saveenv»? …







localhost ~ # ls -l /boot/uboot.env
-rwxr-xr-x 1 root root 8192 Dec  2 13:22 /boot/uboot.env
      
      





, . (, ). , ?







localhost ~ # mv /boot/uboot.env /boot/uboot.env.bak
localhost ~ # fw_printenv
Cannot open /boot/uboot.env: No such file or directory
localhost ~ # mv /boot/uboot.env.bak /boot/uboot.env
      
      





. . , … , . . , . ? 950 tools/env/fw_env.c:







        lseek(fd, blockstart + block_seek, SEEK_SET);

        rc = read(fd, buf + processed, readlen);
        if (rc == -1) {
            fprintf(stderr, "Read error on %s: %s\n",
                DEVNAME(dev), strerror(errno));
            return -1;
        }
        if (rc != readlen) {
            fprintf(stderr,
                "Read error on %s: Attempted to read %zd bytes but got %d\n",
                DEVNAME(dev), readlen, rc);
            return -1;
        }
      
      





. read(). . , , read() -1, errno . . ? , … …







, read? , … read- -. read() . . ? , — .







localhost ~ # strace fw_printenv
execve("/usr/bin/fw_printenv", ["fw_printenv"], 0x7ebf2400 /* 28 vars */) = 0
brk(NULL)                               = 0x2118000
uname({sysname="Linux", nodename="localhost", ...}) = 0
access("/etc/ld.so.preload", R_OK)      = -1 ENOENT (No such file or directory)
openat(AT_FDCWD, "/etc/ld.so.cache", O_RDONLY|O_LARGEFILE|O_CLOEXEC) = 3
fstat64(3, {st_mode=S_IFREG|0644, st_size=42265, ...}) = 0
mmap2(NULL, 42265, PROT_READ, MAP_PRIVATE, 3, 0) = 0x76f14000
close(3)                                = 0
openat(AT_FDCWD, "/lib/libc.so.6", O_RDONLY|O_LARGEFILE|O_CLOEXEC) = 3
read(3, "\177ELF\1\1\1\3\0\0\0\0\0\0\0\0\3\0(\0\1\0\0\0\f~\1\0004\0\0\0"..., 512) = 512
fstat64(3, {st_mode=S_IFREG|0755, st_size=1286448, ...}) = 0
mmap2(NULL, 8192, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x76f12000
mmap2(NULL, 1356160, PROT_READ|PROT_EXEC, MAP_PRIVATE|MAP_DENYWRITE, 3, 0) = 0x76da1000
mprotect(0x76ed7000, 65536, PROT_NONE)  = 0
mmap2(0x76ee7000, 12288, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED|MAP_DENYWRITE, 3, 0x136000) = 0x76ee7000
mmap2(0x76eea000, 8576, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED|MAP_ANONYMOUS, -1, 0) = 0x76eea000
close(3)                                = 0
set_tls(0x76f12ca0)                     = 0
mprotect(0x76ee7000, 8192, PROT_READ)   = 0
mprotect(0x4a9000, 4096, PROT_READ)     = 0
mprotect(0x76f1f000, 4096, PROT_READ)   = 0
munmap(0x76f14000, 42265)               = 0
openat(AT_FDCWD, "/var/lock/fw_printenv.lock", O_WRONLY|O_CREAT|O_TRUNC, 0666) = 3
flock(3, LOCK_EX)                       = 0
brk(NULL)                               = 0x2118000
brk(0x2139000)                          = 0x2139000
openat(AT_FDCWD, "/etc/fw_env.config", O_RDONLY) = 4
fstat64(4, {st_mode=S_IFREG|0644, st_size=1342, ...}) = 0
read(4, "# Configuration file for fw_(pri"..., 4096) = 1342
read(4, "", 4096)                       = 0
close(4)                                = 0
openat(AT_FDCWD, "/boot/uboot.env", O_RDONLY) = 4
fstat64(4, {st_mode=S_IFREG|0755, st_size=8192, ...}) = 0
close(4)                                = 0
openat(AT_FDCWD, "/boot/uboot.env", O_RDONLY) = 4
_llseek(4, 0, [0], SEEK_SET)            = 0
read(4, "n.'\202__INF0__=Ravion-V2 I.MX6 CPU"..., 16384) = 8192
write(2, "Read error on /boot/uboot.env: S"..., 39Read error on /boot/uboot.env: Success
) = 39
close(4)                                = 0
flock(3, LOCK_UN)                       = 0
close(3)                                = 0
exit_group(1)                           = ?
+++ exited with 1 +++
localhost ~ #
      
      





Linux. . . , — . -. :







openat(AT_FDCWD, "/boot/uboot.env", O_RDONLY) = 4
_llseek(4, 0, [0], SEEK_SET)            = 0
read(4, "n.'\202__INF0__=Ravion-V2 I.MX6 CPU"..., 16384) = 8192
write(2, "Read error on /boot/uboot.env: S"..., 39Read error on /boot/uboot.env: Success
) = 39
      
      





16384 (16K), 8192 (8K). . . . , 8192 . . 0, 0x4000 16384. 0x2000







# VFAT example
/boot/uboot.env 0x0000          0x2000
      
      





, . U-Boot’ . . . . () . — - . , . , . . .







— U-Boot. . , — . ? ( )? — 16 8. — ? , — .







localhost ~ # fw_printenv
__INF0__=Ravion-V2 I.MX6 CPU Module BSP package
__INF1__=Created: Alex A. Mihaylov AKA MinimumLaw, MinimumLaw@Rambler.Ru
[…]
boot_os=1
localhost ~ #
      
      





. fw_setenv .







localhost ~ # fw_setenv boot_os 0; fw_printenv boot_os
boot_os=0
      
      





? . , . , ?







. , U-Boot, , . . , strace read 8192. ? 8192 -1.







. , — , Das U-Boot . . , . . , . . . .







localhost ~ # fw_printenv --version
Compiled with U-Boot 2019.10
localhost ~ #
      
      





! (2020.10). . OpenSource . .







        lseek(fd, blockstart + block_seek, SEEK_SET);

        rc = read(fd, buf + processed, readlen);
        if (rc != readlen) {
            fprintf(stderr, "Read error on %s: %s\n",
                DEVNAME(dev), strerror(errno));
            return -1;
        }

      
      





. . . . . , . .







. «uboot.env»







localhost ~ # hexdump -C /boot/uboot.env
00000000  0a 43 62 eb 5f 5f 49 4e  46 30 5f 5f 3d 52 61 76  |.Cb.__INF0__=Rav|
00000010  69 6f 6e 2d 56 32 20 49  2e 4d 58 36 20 43 50 55  |ion-V2 I.MX6 CPU|
00000020  20 4d 6f 64 75 6c 65 20  42 53 50 20 70 61 63 6b  | Module BSP pack|
00000030  61 67 65 00 5f 5f 49 4e  46 31 5f 5f 3d 43 72 65  |age.__INF1__=Cre|
[...]
00000720  3d 71 70 00 76 65 6e 64  6f 72 3d 72 61 76 69 6f  |=qp.vendor=ravio|
00000730  6e 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |n...............|
00000740  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
00002000
localhost ~ #
      
      





— , , . 1837 (0x7031 – 4) . 4 CRC32, =. . . . ( !) . ?







. . U-Boot . vfat . . OpenWRT . SPI-flash. . . . . , dataflash raw-NAND . .. , . .







. … . . . . , . . , . .







. , … , . : « , . .» , .







P.S.

CodeRushشكرا مرة أخرى على الدعوة إلى هبر. ونعم ، أريد دائمًا أن أكتب عن شيء جاد - حول المجمعين ، حول البرمجة الآمنة مباشرة على الأجهزة. والقوة كافية فقط لقراءة الجمعة الخفيفة. حسنًا ، لنفترض أنه قد تم البدء. تبدأ الرحلة الكبيرة دائمًا بخطوة صغيرة.








All Articles