1.8 編寫一個統(tǒng)計空格密强、制表符、換行符的程序
統(tǒng)計這些符號至少需要一個文件
所以:需要先打開一個文件(傳入-- 打開 讀取 文件位置 權(quán)限等參數(shù))
open_fd_in:
movl $5,%eax
movl 8(%ebp),%ebx
movl $0,%ecx
movl $0666,%edx
int $0x80
然后:得到這個文件的標(biāo)識符
movl %eax,-4(%ebp)
然后:讀取這個文件(傳入-- 讀取 文件標(biāo)識符 緩沖區(qū) 緩沖區(qū)空間 等參數(shù))
movl $3,%eax
movl -4(%ebp),%ebx
movl $BUFFER_DATA,%ecx
movl $500,%edx
int $0x80
cmpl $0 , %eax
然后:從緩沖區(qū)將信息寫入另一文件猪半。
movl %eax,%edx
movl $4,%eax
movl -8(%ebp),%ebx
movl $BUFFER_DATA,%ecx
int $0x80
以下程序陷入死循環(huán)催什,不知道問題,用GDB檢查
int main(int argc,char *argv[]){
int fd_in,fd_out;
int readnum,writenum;
int flag=0;
fd_in=open(argv[1] , O_RDONLY,0);//fd_in中接受到文件的標(biāo)識號鸦列,在32位系統(tǒng)上妥色,他是一個4字節(jié)的int值;
fd_out=open(argv[2],O_WRONLY,0);//同上搪花,只不過這次打開的是將要寫入的文件。
char buf[BUFSIZ];
while (flag!=1){
readnum=read(fd_in,buf,BUFSIZ);
if (readnum=0)
{flag=1;}
else if(readnum=-1)
{}
else if(readnum<-1)
{}
else {
write(fd_out,buf,readnum);
}
}
return 0;
}
(gdb) display
(gdb) s
open () at ../sysdeps/unix/syscall-template.S:81
81 ../sysdeps/unix/syscall-template.S: No such file or directory.
找到了問題嘹害,原因是找不到open(),這個函數(shù)需要include <fcntl.h>,加上再試:
依然是:
(gdb) s
open () at ../sysdeps/unix/syscall-template.S:81
81 ../sysdeps/unix/syscall-template.S: No such file or directory.
去掉 include <stdio.h>再試發(fā)現(xiàn)有一個syscalls.h的頭文件我沒有包含撮竿,但是我的確沒有這個頭文件。
算了笔呀,我自己寫一個open函數(shù)好了幢踏。
//open.c
#include <stdio.h>
int main(int argc,char *argv[]){
int fd_in;
fd_in=open(argv[1] ,0,0);//fd_in中接受到文件的標(biāo)識號,在32位系統(tǒng)上凿可,他是一個4字節(jié)的int值;
printf("Hello,world! %d",fd_in);
return fd_in;
}
~
~
~
將.c文件編譯成匯編文件
gcc -S open.c -o open1.s
.file "open.c"
.text
.globl _start
.type _start, @function
_start:
.LFB0:
.cfi_startproc
pushl %ebp
.cfi_def_cfa_offset 8
.cfi_offset 5, -8
movl %esp, %ebp
.cfi_def_cfa_register 5
andl $-16, %esp
subl $32, %esp
movl 12(%ebp), %eax
addl $4, %eax
movl (%eax), %eax
movl $0, 8(%esp)
movl $0, 4(%esp)
movl %eax, (%esp)
call open
movl %eax, 28(%esp)
movl 28(%esp), %eax
leave
.cfi_restore 5
.cfi_def_cfa 4, 4
ret
.cfi_endproc
.LFE0:
.size _start, .-_start
.ident "GCC: (Ubuntu 4.8.4-2ubuntu1~14.04.4) 4.8.4"
.section .note.GNU-stack,"",@progbits
修改main為_start,(后注:這個方式會出現(xiàn)段錯誤惑折,運(yùn)行到_start時便會出錯,感覺是因?yàn)閙ain是函數(shù)枯跑,而_start替代main時也成為了函數(shù))
as open1.s -o open1.o
然后編寫open匯編函數(shù)
.globl open
.type open,@function
open:
pushl %ebp
movl %esp,%ebp
movl $5,%eax
movl 8(%ebp),%ebx
movl 12(%ebp),%ecx
movl $0666,%edx
int $0x80
store_fd_in:
movl %eax,-4(%ebp)
movl %ebp,%esp
pop %ebp
ret
as asmopen.s -o asmopen.o
鏈接:
root@192:~/workdir/KandG/1# ld open1.o asmopen.o -o target
root@192:~/workdir/KandG/1# ./target in.txt
Segmentation fault (core dumped)
以下方法解決惨驶,open函數(shù)使用成功
root@192:~/workdir/KandG/1# gcc -c open.c -o open2.o
root@192:~/workdir/KandG/1# as asmopen.s -o asmopen.o
root@192:~/workdir/KandG/1# gcc open2.o asmopen.o -o target1
root@192:~/workdir/KandG/1# ./target1 ./in.txt
Hello,world! 3root@192:~/workdir/KandG/1#
nnYJ