可以看看dd命令的实现,在busybox里有
补充:
/dev/mtd/* 或者 /dev/mtd* 这两种表示方式一般表示的是字符设备/dev/mtdblock/* 或者 /dev/mtdblock* 这两种是块设备的表示方式mount 的一般都是块设备貌似dd只对字符设备进行 *** 作
举例:
The mtd0 is char device of mtdblock0.
Create char mtd devices >
>mknod /dev/mtd0 c 90 0
>mknod /dev/mtd1 c 90 2
Linux系统中/dev/mtd与/dev/mtdblock的区别,即MTD字符设备和块设备的区别
1. /dev/mtdN 是Linux 中的MTD架构中,系统自己实现的mtd分区所对应的字符设备,其里面添加了一些ioctl,支持很多命令,如MEMGETINFO,MEMERASE等。而mtd-util中的flash_eraseall等工具,就是以这些ioctl为基础而实现的工具,实现一些关于Flash的 *** 作。比如,mtd 工具中的 flash_eraseall中的:if (ioctl(fd, MEMGETINFO, &meminfo) != 0) { fprintf(stderr, "%s: %s: unable to get MTD device info\n", exe_name, mtd_device) return 1}其中,MEMGETINFO,就是Linux MTD中的drivers/mtd/nand/mtdchar.c中的:static int mtd_ioctl(struct inode *inode, struct file *file, u_int cmd, u_long arg){。。。。。case MEMGETINFO: info.type = mtd->type info.flags = mtd->flags info.size = mtd->size info.erasesize = mtd->erasesize info.writesize = mtd->writesize info.oobsize = mtd->oobsize /* The below fields are obsolete */ info.ecctype = -1 info.eccsize = 0 if (copy_to_user(argp, &info, sizeof(struct mtd_info_user)))return -EFAULT break。。。}而/dev/mtdblockN,是Nand Flash驱动中,驱动在用add_mtd_partitions()添加MTD设备分区,而生成的对应的块设备。根据以上内容,也就更加明白,为什么不能用nandwrite,flash_eraseall,flash_erase等工具去对/dev/mtdblockN去 *** 作了。因为/dev/mtdblock中不包含对应的ioctl,不支持你这么 *** 作。2. mtd char 设备的主设备号是90,而mtd block设备的主设备号是31:# ls /dev/mtd? -l crw-r-----1 root root 90, 0 May 30 2007 /dev/mtd0crw-r-----1 root root 90, 2 May 30 2007 /dev/mtd1crw-r-----1 root root 90, 4 Jul 17 2009 /dev/mtd2crw-r-----1 root root 90, 6 May 30 2007 /dev/mtd3crwxrwxrwx1 root root 90, 8 May 30 2007 /dev/mtd4crwxrwxrwx1 root root 90, 10 May 30 2007 /dev/mtd5crwxrwxrwx1 root root 90, 12 May 30 2007 /dev/mtd6crwxrwxrwx1 root root 90, 14 May 30 2007 /dev/mtd7crwxrwxrwx1 root root 90, 16 May 30 2007 /dev/mtd8crwxrwxrwx1 root root 90, 18 May 30 2007 /dev/mtd9# ls /dev/mtdblock? -lbrw-r-----1 root root 31, 0 May 30 2007 /dev/mtdblock0brw-r-----1 root root 31, 1 May 30 2007 /dev/mtdblock1brw-r-----1 root root 31, 2 May 30 2007 /dev/mtdblock2brw-r-----1 root root 31, 3 May 30 2007 /dev/mtdblock3brwxrwxrwx1 root root 31, 4 May 30 2007 /dev/mtdblock4brwxrwxrwx1 root root 31, 5 May 30 2007 /dev/mtdblock5brwxrwxrwx1 root root 31, 6 May 30 2007 /dev/mtdblock6brwxrwxrwx1 root root 31, 7 May 30 2007 /dev/mtdblock7brwxrwxrwx1 root root 31, 8 May 30 2007 /dev/mtdblock8brwxrwxrwx1 root root 31, 9 May 30 2007 /dev/mtdblock9此设备号,定义在/include/linux/mtd/mtd.h中 :#define MTD_CHAR_MAJOR 90#define MTD_BLOCK_MAJOR 313. 其中,mtd的块设备的大小,可以通过查看分区信息获得:# cat /proc/partitionsmajor minor #blocks name31 0 1024 mtdblock031 1 8192 mtdblock131 2 204800 mtdblock231 3 65536 mtdblock331 4 225280 mtdblock4上面中显示的块设备大小,是block的数目,每个block是1KB。而每个字符设备,其实就是对应着上面的每个块设备。即/dev/mtd0对应/dev/mtdblock0,其他以此类推。换句话说,mtdblockN的一些属性,也就是mtdN的属性,比如大小。4。对每个mtd字符设备的 *** 作,比如利用nandwrite去对/dev/mtd0写数据,实际就是 *** 作/dev/mtdblock0。而这些 *** 作里面涉及到的偏移量offset,都指的是此mtd 分区内的偏移。比如向/dev/mtd1的offset为0的位置写入数据,实际 *** 作的是物理偏移offset=/dev/mtd0的大小=1MB=0x100000。
【具体步骤】:1) 下载 busybox 源代码,并解包
$ wget -c http //www点busybox点net/downloads/busybox-1.7.0.tar.bz2
$ tar jxvf busybox-1.7.0.tar.bz2
2) 下载交叉编译工具,并安装
我下载的是: arm-2009q1-161-arm-none-eabi.bin
说明:要正确设置好 PATH 变量。
例如将 “ 你的目录 ”/CodeSourcery/Sourcery_G++_Lite/bin 加到 PATH 路径中。
3 )进入到 busybox 解压后的源文件目录中,修改 Makefile
将第 176 行改为:
CROSS_COMPILE ?=arm-none-linux-gnueabi-
4 )进行编译选项配置
a 、
$ make menuconfig
Busybox Settings --->
Build Options --->
[*] Build BusyBox as a static binary(no shared libs)
说明:这个选项一定要选,这样才能把 busybox 编译成静态链接的可执行文件,运行时可以独立于其他库。
b 、
Installation Options --->
[*] Don't use /usr
说明:这个也一定要选,否则 make install 后, busybox 将安装在原来系统的 /usr 下,将你原有的命令都覆盖了!
5 )配置好后可以编译了,执行如下命令:
$ make
发现没过多久,就报错了,晕,错误内容如下:
applets/applets.c:20:2: warning: #warning Static linking against glibc produces buggy executables
applets/applets.c:21:2: warning: #warning (glibc does not cope well with ld --gc-sections).
applets/applets.c:22:2: warning: #warning See sources.redhat.com/bugzilla/show_bug.cgi?id=3400
applets/applets.c:23:2: warning: #warning Note that glibc is unsuitable for static linking anyway.
applets/applets.c:24:2: warning: #warning If you still want to do it, remove -Wl,--gc-sections
applets/applets.c:25:2: warning: #warning from top-level Makefile and remove this warning.
applets/applets.c:26:2: error: #error Aborting compilation.
make[1]: *** [applets/applets.o] 错误 1
make: *** [applets] 错误 2
看到它给出了提示,说 glibc 库不适和用来静态编译,最后给出解决方案就是将 applets/applets.c 中这部分内容给去掉,也就是 19-27 行。
然后再 make 进行编译。
不多久又报错了,看看具体错误:
.../compal/CodeSourcery/Sourcery_G++_Lite/bin/../arm-none-linux-gnueabi/libc/usr/include/linux/netfilter.h:56:17:
error: field 'in' has incomplete type
.../CodeSourcery
/Sourcery_G++_Lite/bin/../arm-none-linux-gnueabi/libc/usr/include/linux/netfilter.h:57:18:
error: field 'in6' has incomplete type
ipsvd/tcpudp.c: In function 'tcpudpsvd_main':
ipsvd/tcpudp.c:314:10: warning: ignoring return value of 'write', declared with attribute warn_unused_result
make[1]: *** [ipsvd/tcpudp.o] 错误 1
make: *** [ipsvd] 错误 2
看到说在我们下载的交叉编译库中有个头文件中的 in 及 in6 类型不对,解决的办法就是:
在 .../arm-none-linux-gnueabi/libc/usr/include/linux/netfilter.h 的开头
添加缺少的头文件:
#include <netinet/in.h>
然后再进行编译。(这次可以安全到最后了,呵呵)
结束后会在当前目录下看到 busybox 这个可执行文件。
6 )编译步骤已完成,下面就将 busybox 这个可执行文件放到 Android 模拟器下去
$ adb push busybox /system/xbin
说明:若是出现什么 read-only file system 等等之类,执行如下命令即可:
$ adb remount
要是老是不行,那就重新启动 adb
$ adb kill-server
$ adb start-server
要是碰到什么内存不足等等,那原因就是你的那个 Android 模拟器是用 Eclipse 打开的。解决办法就是,手动启动你的 Android emulator 模拟器,具体如下:
$ android list avd # 注释:列出你所有的模拟器
$ emulator -avd your_emulator_name -partition-size 256
注意:最好放在 /system/xbin 下面,这样当你在模拟器里使用 busybox 时不需要指定绝对路径了,否则的话谁都会疯掉嘛,呵呵。
7 )输入 adb shell 进入终端,执行看看
# busybox --help
BusyBox v1.7.0 (2011-04-22 20:53:21 CST) multi-call binary
Copyright (C) 1998-2006 Erik Andersen, Rob Landley, and others.
Licensed under GPLv2. See source distribution for full notice.
…
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)