Shell脚本 - 分段错误

时间:2011-12-20 00:50:10

标签: shell segmentation-fault

我试图让shell脚本正常工作时遇到了一些困难。我正在尝试创建一个脚本,从数据库中读取一行并格式化为文本文件,以便与另一个软件一起使用。脚本需要循环大约5000万次(大型数据库),它只需要5,500到5,800次迭代,然后就会产生分段错误。 我试图用strace跟踪错误(最后几行在下面),但我不确定我在看什么。

clone(child_stack=0,flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD,                      child_tidptr=0xb76f8728) = 17547
close(4)                                = 0
close(5)                                = 0
pipe([4, 5])                            = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD,     child_tidptr=0xb76f8728) = 17548
close(3)                                = 0
close(5)                                = 0
pipe([3, 5])                            = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD, child_tidptr=0xb76f8728) = 17549
close(4)                                = 0
close(5)                                = 0
pipe([4, 5])                            = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD, child_tidptr=0xb76f8728) = 17550
close(3)                                = 0
close(5)                                = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD, child_tidptr=0xb76f8728) = 17551
close(4)                                = 0
close(-1)                               = -1 EBADF (Bad file descriptor)
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17546
--- SIGCHLD (Child exited) @ 0 (0) ---
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17547
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17548
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17549
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17551
--- SIGCHLD (Child exited) @ 0 (0) ---
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17550
--- SIGCHLD (Child exited) @ 0 (0) ---
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17545
--- SIGCHLD (Child exited) @ 0 (0) ---
write(1, "OK!\n", 4OK!
)                    = 4
pipe([3, 4])                            = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD,   child_tidptr=0xb76f8728) = 17552
close(4)                                = 0
read(3, "10632\n", 128)                 = 6
read(3, "", 128)                        = 0
close(3)                                = 0
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17552
--- SIGCHLD (Child exited) @ 0 (0) ---
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD,    child_tidptr=0xb76f8728) = 17553
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17553
--- SIGCHLD (Child exited) @ 0 (0) ---
write(1, "Preparing file 10632 of ", 24Preparing file 10632 of ) = 24
write(1, "51041073(7) ....", 1651041073(7) ....)        = 16
pipe([3, 4])                            = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD,    child_tidptr=0xb76f8728) = 17554
close(4)                                = 0
read(3, "8\n", 128)                     = 2
read(3, "", 128)                        = 0
close(3)                                = 0
wait4(-1, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 17554
--- SIGCHLD (Child exited) @ 0 (0) ---
pipe([3, 4])                            = 0
clone(child_stack=0, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD, child_tidptr=0xb76f8728) = 17555
close(4)                                = 0
read(3, "", 128)                        = 0
close(3)                                = 0
wait4(-1, [{WIFSIGNALED(s) && WTERMSIG(s) == SIGSEGV}], 0, NULL) = 17555
--- SIGCHLD (Child exited) @ 0 (0) ---
--- SIGSEGV (Segmentation fault) @ 0 (0) ---
+++ killed by SIGSEGV +++
Segmentation fault

脚本如下:

#!/bin/sh
#----VARS----
prefix="10.0.3.2/read/"
oprdir="/home/andrew/doc/read/"
throttle=0

#----------------
x=5000
advcount=0
count=`curl "$prefix""count.php" 2> /dev/null`

#----Funcs----
getline()
{
  #Run curl to get the line of text
  #TODO -n
  echo -n "Preparing file" $x "of "
  if [ $advcount -ge 25 ]
  then
    count=`curl "$prefix""count.php" 2> /dev/null`
    advcount=0
  fi
  echo -n $count"("$advcount")" "...."
  advcount=`expr $advcount + 1`

  line=`curl "$prefix""testfile-prep.php?x=$x" 2> /dev/null`
  if [ "$line" = "ERR ERROR: X OUTSIDE RECORDS." ]
  then
    echo "ERROR: X OUTSIDE DB... WAITING TO RETRY."
    sleep 60
    getline
  fi
  prepline
}

prepline()
{
  echo $line | sed 's/^[0-9]*\./\n./g' | sed 's/\([a-zA-Z]*\)\./\1\n./g'  | sed 's/\,/\n\,/g' | sed 's/(\(.*\)/(\n\1/g' | sed 's/\(.*\))/\1\n)/g' | sed 's/ /\n/g'> out-0-$x.dat
  echo "OK!"
  #cat out-$x.dat
  advance
}
advance()
{
  x=`expr $x + 1`
  sleep $throttle
  getline
}

cd "$oprdir"
getline

对于novelette的信息感到抱歉,任何帮助表示赞赏。

1 个答案:

答案 0 :(得分:4)

您有stack overflow

您的getline函数调用prepline,调用advance,调用getline

你应该使用某种循环而不是在非异常条件下调用函数。 (错误情况下的自我调用是可以的,你不会有5,000个错误溢出堆栈。)