目标系统必须包括gdbserver程序,/etc/shells命令查看
分类:巴黎人-服务器

前言

这周在学习linux脚本编程,调试时,用echo 或bash -x
检查语法用bash -n x.sh

在函数中,要用echo返回出参字符串,如果再用echo打印调试信息,将逻辑搞乱了。
当时就在想,linux工具这么强大,既然有gdb, 那就有脚本的调试器才对啊.

今天在看书时,看到了bashdb这个工具。
在debian下试过了,好使。
虽然没有gdb -tui 那么好使,也满意了。

bashdb中也有好多调试命令,进入bashdb后,用help命令可以看到命令列表。
对于脚本的初级调试,掌握几个命令就够用。

    exit 1

一.前言
嵌入式Linux系统中,应用开发过程中,很多情况下,用户需要对一个应用程序进行反复调试,
特别是复杂的程序。采用GDB方法调试,由于嵌入式系统资源有限性,一般不能直接在目标系统上进行调试,
通常采用gdb+gdbserver的方式进行调试。 Gdbserver在目标系统中运行,gdb则在宿主机上运行。
目标系统必须包括gdbserver程序,宿主机也必须安装gdb程序。在此我们还不能直接采用linux
发行版自带的gdb,需要交叉编译gdb和gdbserver。

$ 除了用于变量前面的标识符外,还有一个妙用,就是和 ‘!’ 结合起来使用。

<5>. 函数 

linux sh : bashdb的安装和初步使用,shbashdb

比如,下面的脚本会在退出时,执行echo:

CC=/opt/toolchain/arm-linux/bin/arm-linux-gcc 
../gdb-6.8/gdb/gdbserver/configure --host=arm-linux --without-included-regex --without-included-gettext

前面已经提过过管道符 “|”, 就是把前面的命令运行的结果丢给后面的命令。

6.2 两个命令

shell脚本执行可以通过./shell-filename.sh的形式执行,另外的一种形式是通过bash ./shell-filename.sh的形式执行,同时在执行脚本时,可以使用-v或者是-x参数打印程序执行信息。

-v:默认打印shell读取的内容

-x:将命令“展开” 之后打印

例如对于下面的程序:

!/bin/sh

# for debug shell script
#
tot=`expr $1 + $2`
echo $tot

如果使用-v选项,结果如下:

 xuqiang@ubuntu:~/shell$ sh -v ./debug.sh 2 5

#!/bin/sh

#

# for debug shell script

#

tot=`expr $1 + $2`

echo $tot

7

如果是使用-x选项的话,结果:

 xuqiang@ubuntu:~/shell$ sh -x ./debug.sh 2 5

+ expr 2 + 5

+ tot=7

+ echo 7

7

下载点

bashdb-4.4-0.94.tar.bz2.7z

[tt.sh : 7] i=0 ?

echo "done"

1.显示环境变量 HOME

<2>. 还是hello world程序 

首先使用vim编辑器(或者是linux下任意的文本编辑器)编写文件helloshell.sh(没有必要使用.sh后缀名):

 #!/bin/bash
echo "hello shell";

 

保存上面的文件,增加该文件的执行权限:

 xuqiang@ubuntu:~/shell$ sudo chmod +x ./helloshell.sh

运行该shell程序:

 xuqiang@ubuntu:~/shell$ ./helloshell.sh

hello shell

通过上面的程序没有什么实际的含义,但是通过第一个shell程序了解shell程序的执行过程。 

实验

# backup /etc/apt/sources.list
cp /etc/apt/sources.list /etc/apt/sources.list.bk1
# add one row on the file(/etc/apt/sources.list)'s end for install bashdb
vi /etc/apt/sources.list
deb http://ftp.de.debian.org/debian wheezy main
wq
# update install source
aptitude update
# install bashdb
aptitude install bashdb
# bash doc
# http://www.gnu.org/software/bash/manual/bashref.html
# debug linux sh
# cd the .sh file dir, debug x.sh by bashdb
bashdb $(pwd)/main.sh
# bashdb's buildin command list
bashdb<0> help
Available commands:
  action     condition  edit     frame    load     run     source  unalias  
  alias      continue   enable   handle   next     search  step    undisplay
  backtrace  debug      eval     help     print    set     step-   untrace  
  break      delete     examine  history  pwd      shell   step+   up       
  clear      disable    export   info     quit     show    tbreak  watch    
  commands   display    file     kill     return   signal  trace   watche   
  complete   down       finish   list     reverse  skip    tty  
# list command
bashdb<1> help list
list[>] [LOC|.|-] [NUMBER] 
LOC is the starting location or dot (.) for current file and
line. Subsequent list commands continue from the last line
listed. Frame switching however resets the line to dot. LOC can be a
read-in function name or a filename and line number separated by a
colon, e.g /etc/profile:5
If NUMBER is omitted, use the LISTSIZE setting as a count. Use "set
listsize" to change this setting. If NUMBER is given and is less than
the starting line, then it is treated as a count. Otherwise it is
treated as an ending line number.
By default aliases "l>" and "list>" are set to list. In this case and
more generally when the alias ends in ">", rather than center lines
around LOC that will be used as the starting point.
Examples:
list .      # List centered around the curent frame line
list        # Same as above if the first time. Else start from where
            # we last left off.
list -      # list backwards from where we left off.
list> .     # list starting from the current frame line.
list  10 3  # list 3 lines centered around 10, lines 9-11
list> 10 3  # list lines 10-12
list  10 13 # list lines 10-13
list  10 -5 # list from lines to 5 lines before teh end of the file
list  /etc/profile:5  # List centered around line 5 of /etc/profile.
list  /etc/profile 5  # Same as above.
list usage  # list centered around function usage().
# 列出脚本源码-当前
l .
# 列出脚本源码-指定行数
l 0
# bashdb的命令和gdb兼容, 用gdb的命令就可以了
# n : 步过
# s : 步入
# 回车 : 执行上一条命令
# 打印变量
print $i_index
pr $i_index
# 指定行数下断点
break 18
# 用debian源提供的bashdb下断点有bug, continue后,断不下来
# 去官网上下载一个新的bashdb : https://sourceforge.net/projects/bashdb/files/bashdb/4.4-0.94/
tar xvf ./bashdb-4.4-0.94.tar.bz2
cd bashdb-4.4-0.94/
aptitude install autotools-dev
make clean
./configure
make
make check # 这步有2项通不过,不影响
# SKIP: test-file-with-spaces
# FAIL: test-sig
# SKIP: test-bug-loc
# FAIL: test-bug-set-e
aptitude remove bashdb
rm -rf /usr/share/bashdb/
make install
# 可以用新版bashdb重新调试脚本了
bashdb $(pwd)/main.sh
# 列出当前文件所有内容, 行数范围设置大点 从0~1000, 如果这个当前sh只有45行,就都显示全了
l 0 1000
# 对指定文件的指定行数下断点
break /home/dev/main.sh:35
# 对函数下断点
break func_show_title
# 继续 continue
c
# 断点列表
L
# 列出指定文件指定行数
l /home/dev/common/sh.prog_version:16
# 删除断点(指定断点号码, 用L命令看断点列表)
delete 1
# bashdb好像没有在调试中,设置变量值的控制语法
# 执行完一个函数或一个循环
finish
# 退出调试器
q
# 好像初步使用bashdb, 查的这些调试命令,已经够用了
# 看一本书上,人家调试就用echo 或 bash -x, 人家对脚本编程该有多熟练啊
# 如果是我,只有在没有条件的情况下才会用echo, 有利器为啥不用呢?

done

../gdb-6.8/configure --target=arm-linux --enable-shared --prefix=/usr/src/arm/gdb/build-gdb --without-x --disable-gdbtk --disable-tui --without-included-regex --without-included-gettext
make

行两个或两个以上的命令如何呢?则需要在命令之间加一个 ”;” 了。

4.3 if-else结构:

图片 1

!/bin/sh#
# see whether arguments is positive
#
if [ $# -ne 1 ]
then
        echo "$0 : you must give/supply one integers"
        exit 1
fi

if test $1 -gt 0
then
        echo "$1 number is postivie"
else
        echo "$1 number is negative"
fi

图片 2

 

图片 3

#!/bin/sh                               

osch=0

echo "1. unix(sun os)"
echo "2. linux(red hat)"
echo -n "select your os choice [1 or 2] ?"
read osch

if [ $osch -eq 1  ]
then
        echo "you pick up unix"
else
        #
        # nested if
        if [ $osch -eq 2 ]
        then
                echo "you pick up linux"
        else
                echo "what you donot like unix/linux"
        fi
fi

图片 4

 

 

图片 5

#!/bin/sh
#
# test the if .. elif .. else
#
if [ $1 -gt 0 ] ; then
        echo "$1 is positive"
elif [ $1 -lt 0 ] ; then
        echo "$1 is negative"
elif [ $1 -eq 0 ] ; then
        echo "$1 is zero"
else
        echo "$1 is not a number"
fi

图片 6

如果想全程打开 xtrace,可以在执行脚本的时候加 -x 参数。

三.gdbserver使用
(1) 目标机中
    执行命令gdbserver 10.0.12.144:1234 test
    注意test在编译的时候是要加-g选项的
(2) 宿主机中
    gdb test
    再输入命令target remote 10.0.12.143:1234
    接下来就可以gdb的调试了
(3) 说明
    目标机IP:10.0.12.143
    宿主机IP: 10.0.12.144
    端口是随意指定的,只要两端保持一致就行了
    test程序两端也必须相同

&& 与 ||,用于多条命令中间的特殊符号

4.1 条件判定

在shell中条件判断是通过test命令或者是[ ]实现, 判断条件如下:

数学运算: 

a -eq b :a == b
a -ne b : a != b
a -lt b : a < b
a -le b : a <= b
a -gt b : a > b
a -ge b : a >= b

string比较:
string1 = string2
string1 != string2
string1 : string1 不是NULL,或者是没有定义

echo this is a test

二.编译gdb和gdbserver工具
下载最新版本的gdb-6.8.tar.gz,可以到ftp://ftp.gnu.org/gnu/gdb下载.
准备好自己的工作目录,如下示
/home/user/gdb
            |--gdb-6.8
            |--obj-gdb
                    |--build
            |--obj-gdbserver
                    |--build
            |--bin
                    
脚本obj-gdb/build内容如下:
#!/bin/sh

## This is first bash shell on centos

<3>. shell中的变量 

sigspec 包括 <signal.h> 中定义的各个 signal, EXIT,ERR,RETURN 和 DEBUG。

脚本obj-gdbserver/build内容如下:
#!/bin/sh

管道符

5.1 函数声明和定义

下面的程序中定义函数demo,向函数传递的所有参数表示为$*,第一个参数$1,第二个参数$2, 依次类推。

图片 7

#!/bin/bash

function demo()
{
        echo "all function args : $*"
        echo "the first arg : $1"
        echo "the second arg : $2"
        echo "the third arg : $3"
}

# call the function

demo -f foo bar  

图片 8

    ((i++))

开始编译:
(1) cd obj-gdb
    ./build
(2) cd obj-gdbserver
    ./build
    这时会生成Makefile,修改Makefile如下
    LDFLAGS= -static
    然后make
(3) cd bin
    cp ../obj-gdb/gdb/gdb .
    cp ../obj-gdbserver/gdbserver .
    arm-linux-strip gdbserver
(4) 好了bin目录下就是最终编译出来的gdb+gdbserver了

个符号就是这里的 2> 和 2>> 分别表示错误重定向和错误追加重定向,当我们运行一

4.2 组合判定 

逻辑运算符:

! : not

exp1 -a exp2 : a && b

exp1 -o exp2 : exp1 || exp2 

ERROR "this is a error log"

echo "done"

单, alias [命令别名]=['具体的命令']

5.2 函数调用(函数参数)

shell中向函数传递参数是通过直接在函数调用后面添加参数完成,在函数中,传递的参数通过$1得到。

打印一些变量,或者提示信息。这应该是一个通用的方法了。在 BASH 里,我们可以简单的用 echo,或者 printf 来输出一些 log,或者加一些 loglevel 来过滤一些 log。

户的家目录中的.bash_history 文件中。有一点需要您知道的是,只有当用户正常退出当前shell 时,在当前 shell 中运行的命令才会保存至.bash_history 文件中。

 4.4 for

图片 9

!/bin/sh

#
# test for
#

for i in 1 2 3 4 5
do
        echo "welcome $i times"
done

图片 10

图片 11

#!/bin/bash

for (( i = 0; i <= 5; i++ ))
do
        echo "welcome $i times"

done  

图片 12

 

图片 13 Tip 1. 注意程序中使用的shell脚本的类型

 

图片 14

#!/bin/bash

for(( i = 1; i <= 5; i++  ))
do
        for(( j = 1; j <= 5; ++j ))
        do
                echo -n "$i"
        done

# print a new line
        echo ""
done

图片 15

[tt.sh : 11] ((i++)) ?

系统默认的 alias 指令也就这几个而已,您也可以自定义您想要的指令别名。alias 语法很简

<6>. 脚本调试 

[tt.sh : 11] ((i++)) ?

这里提到的后面的命令,并不是所有的命令都可以的,一般针对文档操作的命令比较常

4.6 case 

图片 16

#!/bin/bash

#
# script to test case statement
#
action="update"
case $action in
        "update")
                echo "update the db"
                ;;
        "select")
                echo "select from db"
                ;;
        "delete")
                echo "delete from db"
                ;;
        *)
                echo "no action"
                ;;

esac  

图片 17

或者,让脚本中命令出错时,把相应的命令打印出来:

境变量只对当前用户起作用。

3.2 用户定义变量

shell中用户可以自定义变量,shell中的变量是没有数据类型的,shell将根据当前的环境自动进行转化,例如:

msg="hello world" 

上面的语句定义变量msg,并设置初始值是为hello world。

1. 需要注意的是定义变量时,=两边是没有空格的 

3.2.1  用户定义变量规则

变量必须是以字母开头,后跟字母或者是下划线,变量的命名是大小写敏感的,并且可以定义一个变量的值为NULL。

xuqiang@ubuntu:~/shell$ vech=

xuqiang@ubuntu:~/shell$ echo $vec

3.2.2  shell中如何使用变量

如果想要得到shell变量中存储的值的话,需要在变量名前增加$符号,例如:

xuqiang@ubuntu:~/shell$ vech="value"

xuqiang@ubuntu:~/shell$ echo $vech # this will print the value of vech

value

xuqiang@ubuntu:~/shell$ echo vech # this will print the string "vech"

vech

3.2.3 全局变量 vs 局部变量 

默认在shell中编写的变量全部是局部变量,如果重新打开console的话,那么这些变量将全部丢失,全局的变量可以写在文件~/.bashrc文件。

是不是有点意思了?其实有一个 bashdb 的开源项目,也是利用 trap 机制,模拟 gdb 做了一个 bash 脚本的调试器。它本身也是一个 bash 脚本。

/etc/bashrc:

6.1 万能的echo

shell的脚本调试是比价恶心的,这里仅仅是提供一些常规性的调试方法,最简单的就是使用echo函数打印出变量的值从而达到调试目的。

trap [-lp] [[arg] sigspec ...]

指令和文件名补全

4.5 while

图片 18

#!/bin/bash

#
# test while loop
#

n=$1
i=0
while [ $i -le 10 ]
do
        echo "$n * $i = `expr $i * $n`"
        i=`expr $i + 1`

done   

图片 19

set +x

Shell 特殊符号

<1>. 什么是shell 

shell扮演者操作系统内核和用户的中间人的角色,用户通过键入shell command,然后shell通过解析用户输入,然后将请求转发给操作系统的内核进行处理。

  1. 一个系统可以存在多个shell,可以通过cat /etc/shells命令查看系统中安装的shell,不同的shell可能支持的命令语法是不相同的。

  2. 可以通过echo $SHELL查看当前使用的shell 

    echo $i

/etc/profile:此文件为系统的每个用户设置环境信息,当用户第一次登录时,该文件被执行.并

<4>. 控制结构 

 ERR: this is a error log

作业控制

3.1 系统变量

linnux下的shell脚本中的变量分为“系统变量”和“用户自定义变量”,可以通过set命令查看那系统变量。

xuqiang@ubuntu:~/shell$ set

... 略去内容

xuqiang@ubuntu:~/shell$ echo $HOME

/home/xuqiang

 

在bash 下,可以使用 * 来匹配零个或多个字符,而用 ? 匹配一个字符。

 

Shell 脚本通常都是以.sh 为后缀名的, 大家一看到.sh 文件就知道这是个 Shell 脚本。helloworld.sh 中第一行要以 “#! /bin/bash” 开头,它代表的意思是,该文件使用的是 bash语法。如果不设置该行,虽然您的shell脚本也可以执行,但是这不符合规范。# 表示注释,在前面讲过的。后面跟一些该脚本的相关注释内容。脚本的注释虽然不是必须的,建议不要省略了。因为随着您写的 shell 脚本越来越多,如果有一天您回头查看自己写过的某个脚本时,很有可能忘记该脚本是用来干什么的以及什么时候写的。所以写上注释是有必要的。而且系统管理员并非只有您一个,如果是其他管理员查看您的脚本,他看不懂岂不是很郁闷.

这个脚本的输出如下:

| 管道符,前面多次出现过,它的作用在于将符号前面命令的结果丢给符号后面的命令。

本文由巴黎人手机版发布于巴黎人-服务器,转载请注明出处:目标系统必须包括gdbserver程序,/etc/shells命令查看

上一篇:没有了 下一篇:没有了
猜你喜欢
热门排行
精彩图文