476 lines
		
	
	
	
		
			9.7 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
			
		
		
	
	
			476 lines
		
	
	
	
		
			9.7 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
#!/bin/sh
 | 
						|
 | 
						|
# this is a line editor using only /bin/sh, /bin/dd and /bin/rm
 | 
						|
 | 
						|
# /bin/rm is not really required, but it is nice to clean up temporary files
 | 
						|
 | 
						|
PATH=
 | 
						|
dd=/bin/dd
 | 
						|
rm=/bin/rm
 | 
						|
 | 
						|
# temporary files we might need
 | 
						|
tmp=/tmp/silly.$$
 | 
						|
ed=/tmp/ed.$$
 | 
						|
trap "$rm -f $tmp $tmp.1 $tmp.2 $tmp.3 $tmp.4 $tmp.5 $tmp.6 $ed.a $ed.b $ed.c; exit" 0 1 2 3
 | 
						|
 | 
						|
# from now on, no more rm - the above trap is enough
 | 
						|
unset rm
 | 
						|
 | 
						|
# we do interesting things with IFS, but better save it...
 | 
						|
saveIFS="$IFS"
 | 
						|
 | 
						|
# in case "echo" is not a shell builtin...
 | 
						|
 | 
						|
Echo () {
 | 
						|
case "$1" in
 | 
						|
  -n) shift
 | 
						|
      $dd of=$tmp 2>/dev/null <<EOF 
 | 
						|
$@
 | 
						|
EOF
 | 
						|
      IFS="+"
 | 
						|
      set `$dd if=$tmp bs=1 of=/dev/null skip=1 2>&1`
 | 
						|
      IFS="$saveIFS"
 | 
						|
      $dd if=$tmp bs=1 count=$1 2>/dev/null
 | 
						|
      ;;
 | 
						|
  *)  $dd 2>/dev/null <<EOF 
 | 
						|
$@
 | 
						|
EOF
 | 
						|
      ;;
 | 
						|
esac
 | 
						|
}
 | 
						|
 | 
						|
# this is used to generate garbage files
 | 
						|
 | 
						|
true () {
 | 
						|
  return 0
 | 
						|
}
 | 
						|
 | 
						|
false () {
 | 
						|
  return 1
 | 
						|
}
 | 
						|
 | 
						|
zero () {
 | 
						|
  ( trap 'go=false' 13
 | 
						|
    go=true
 | 
						|
    while $go
 | 
						|
    do
 | 
						|
      $dd "if=$0"
 | 
						|
      case "$?" in
 | 
						|
	0) ;;
 | 
						|
	*) go=false ;;
 | 
						|
      esac
 | 
						|
    done
 | 
						|
  ) 2>/dev/null
 | 
						|
}
 | 
						|
 | 
						|
# arithmetic using dd!
 | 
						|
 | 
						|
# add variable n1 n2 n3...
 | 
						|
# assigns n1+n2+n3+... to variable
 | 
						|
 | 
						|
add () {
 | 
						|
  result="$1"
 | 
						|
  shift
 | 
						|
  $dd if=/dev/null of=$tmp bs=1 2>/dev/null
 | 
						|
  for n in "$@"
 | 
						|
  do
 | 
						|
    case "$n" in
 | 
						|
      0) ;;
 | 
						|
      *) zero | $dd of=$tmp.1 bs=1 "count=$n" 2>/dev/null
 | 
						|
	 ( $dd if=$tmp; $dd if=$tmp.1 ) 2>/dev/null | $dd of=$tmp.2 2>/dev/null
 | 
						|
	 $dd if=$tmp.2 of=$tmp 2>/dev/null
 | 
						|
	 ;;
 | 
						|
    esac
 | 
						|
  done
 | 
						|
  IFS="+"
 | 
						|
  set `$dd if=$tmp bs=1 of=/dev/null 2>&1`
 | 
						|
  IFS="$saveIFS"
 | 
						|
  eval $result='$1'
 | 
						|
}
 | 
						|
 | 
						|
# subtract variable n1 n2
 | 
						|
# subtracts n2 from n1, assigns result to variable
 | 
						|
 | 
						|
subtract () {
 | 
						|
  result="$1"
 | 
						|
  zero | $dd of=$tmp bs=1 "count=$2" 2>/dev/null
 | 
						|
  IFS="+"
 | 
						|
  set `$dd if=$tmp bs=1 of=/dev/null "skip=$3" 2>&1`
 | 
						|
  IFS="$saveIFS"
 | 
						|
  case "$1" in
 | 
						|
    dd*) set 0 ;;
 | 
						|
  esac
 | 
						|
  eval $result='$1'
 | 
						|
}
 | 
						|
 | 
						|
# multiply variable n1 n2
 | 
						|
# variable = n1 * n2
 | 
						|
 | 
						|
multiply () {
 | 
						|
  result="$1"
 | 
						|
  zero | $dd "bs=$2" of=$tmp "count=$3" 2>/dev/null
 | 
						|
  IFS="+"
 | 
						|
  set `$dd if=$tmp bs=1 of=/dev/null 2>&1`
 | 
						|
  IFS="$saveIFS"
 | 
						|
  eval $result='$1'
 | 
						|
}
 | 
						|
 | 
						|
# divide variable n1 n2
 | 
						|
# variable = int( n1 / n2 )
 | 
						|
 | 
						|
divide () {
 | 
						|
  result="$1"
 | 
						|
  zero | $dd bs=1 of=$tmp "count=$2" 2>/dev/null
 | 
						|
  IFS="+"
 | 
						|
  set `$dd if=$tmp "bs=$3" of=/dev/null 2>&1`
 | 
						|
  IFS="$saveIFS"
 | 
						|
  eval $result='$1'
 | 
						|
}
 | 
						|
 | 
						|
# compare variable n1 n2 sets variable to lt if n1<n2, gt if n1>n2, eq if n1==n2
 | 
						|
 | 
						|
compare () {
 | 
						|
  res="$1"
 | 
						|
  n1="$2"
 | 
						|
  n2="$3"
 | 
						|
  subtract somename "$n1" "$n2"
 | 
						|
  case "$somename" in
 | 
						|
    0) ;;
 | 
						|
    *) eval $res=gt; return;
 | 
						|
  esac
 | 
						|
  subtract somename "$n2" "$n1"
 | 
						|
  case "$somename" in
 | 
						|
    0) ;;
 | 
						|
    *) eval $res=lt; return;
 | 
						|
  esac
 | 
						|
  eval $res=eq
 | 
						|
}
 | 
						|
 | 
						|
# lt n1 n2 returns true if n1 < n2
 | 
						|
 | 
						|
lt () {
 | 
						|
  n1="$1"
 | 
						|
  n2="$2"
 | 
						|
  subtract somename "$n2" "$n1"
 | 
						|
  case "$somename" in
 | 
						|
    0) return 1 ;;
 | 
						|
  esac
 | 
						|
  return 0
 | 
						|
}
 | 
						|
 | 
						|
# le n1 n2 returns true if n1 <= n2
 | 
						|
 | 
						|
le () {
 | 
						|
  n1="$1"
 | 
						|
  n2="$2"
 | 
						|
  subtract somename "$n1" "$n2"
 | 
						|
  case "$somename" in
 | 
						|
    0) return 0 ;;
 | 
						|
  esac
 | 
						|
  return 1
 | 
						|
}
 | 
						|
 | 
						|
# gt n1 n2 returns true if n1 > n2
 | 
						|
 | 
						|
gt () {
 | 
						|
  n1="$1"
 | 
						|
  n2="$2"
 | 
						|
  subtract somename "$n1" "$n2"
 | 
						|
  case "$somename" in
 | 
						|
    0) return 1 ;;
 | 
						|
  esac
 | 
						|
  return 0
 | 
						|
}
 | 
						|
 | 
						|
# ge n1 n2 returns true if n1 >= n2
 | 
						|
 | 
						|
ge () {
 | 
						|
  n1="$1"
 | 
						|
  n2="$2"
 | 
						|
  subtract somename "$n2" "$n1"
 | 
						|
  case "$somename" in
 | 
						|
    0) return 0 ;;
 | 
						|
  esac
 | 
						|
  return 1
 | 
						|
}
 | 
						|
 | 
						|
# useful functions for the line editor
 | 
						|
 | 
						|
# open a file - copy it to the buffers
 | 
						|
 | 
						|
open () {
 | 
						|
  file="$1"
 | 
						|
  set `$dd "if=$file" of=/dev/null 2>&1`
 | 
						|
  case "$1" in
 | 
						|
    dd*) return 1
 | 
						|
  esac
 | 
						|
  # copy the first line to $ed.c
 | 
						|
  go=true
 | 
						|
  len=0
 | 
						|
  while $go
 | 
						|
  do
 | 
						|
    case "`$dd "if=$file" bs=1 skip=$len count=1 2>/dev/null`" in
 | 
						|
      ?*) go=true ;;
 | 
						|
      *) go=false ;;
 | 
						|
    esac
 | 
						|
    add len 1 $len
 | 
						|
  done
 | 
						|
  # now $len is the length of the first line (including newline)
 | 
						|
  $dd "if=$file" bs=1 count=$len of=$ed.c 2>/dev/null
 | 
						|
  $dd "if=$file" bs=1 skip=$len of=$ed.b 2>/dev/null
 | 
						|
  $dd if=/dev/null of=$ed.a 2>/dev/null
 | 
						|
  lineno=1
 | 
						|
}
 | 
						|
 | 
						|
# save a file - copy the buffers to the file
 | 
						|
 | 
						|
save () {
 | 
						|
  # make a backup copy of the original
 | 
						|
  $dd "if=$1" "of=$1.bak" 2>/dev/null
 | 
						|
  # and save
 | 
						|
  ( $dd if=$ed.a; $dd if=$ed.c; $dd if=$ed.b ) > "$1" 2>/dev/null
 | 
						|
}
 | 
						|
 | 
						|
# replace n1 n2 bla replaces n2 chars of current line, starting n1-th
 | 
						|
 | 
						|
replace () {
 | 
						|
  $dd if=$ed.c of=$tmp.1 bs=1 "count=$1" 2>/dev/null
 | 
						|
  ( $dd if=$ed.c "skip=$1" bs=1 | $dd of=$tmp.2 bs=1 "skip=$2" ) 2>/dev/null
 | 
						|
  shift
 | 
						|
  shift
 | 
						|
  ( $dd if=$tmp.1; Echo -n "$@"; $dd if=$tmp.2 ) > $tmp.3 2>/dev/null
 | 
						|
  $dd if=$tmp.3 of=$ed.c 2>/dev/null
 | 
						|
}
 | 
						|
 | 
						|
# rstring n s bla
 | 
						|
# replace the n-th occurence of s with bla
 | 
						|
 | 
						|
rstring () {
 | 
						|
  n="$1"
 | 
						|
  shift;
 | 
						|
  # first we have to find it - this is fun!
 | 
						|
  # we have $tmp.4 => text before string, $tmp.5 => text after
 | 
						|
  $dd if=/dev/null of=$tmp.4 2>/dev/null
 | 
						|
  $dd if=$ed.c of=$tmp.5 2>/dev/null
 | 
						|
  string="$1"
 | 
						|
  shift
 | 
						|
  $dd of=$tmp.6 2>/dev/null <<EOF
 | 
						|
$@
 | 
						|
EOF
 | 
						|
  while :
 | 
						|
  do
 | 
						|
    case "`$dd if=$tmp.5 2>/dev/null`" in
 | 
						|
      $string*)
 | 
						|
	  if lt $n 2
 | 
						|
	  then
 | 
						|
	    # now we want to replace the string
 | 
						|
	    Echo -n "$@" > $tmp.2
 | 
						|
	    Echo -n "$string" > $tmp.1
 | 
						|
	    IFS="+"
 | 
						|
	    set `$dd bs=1 if=$tmp.1 of=/dev/null 2>&1`
 | 
						|
	    IFS="$saveIFS"
 | 
						|
	    slen=$1
 | 
						|
	    IFS="+"
 | 
						|
	    ( $dd if=$tmp.4; $dd if=$tmp.2; $dd if=$tmp.5 bs=1 skip=$slen ) \
 | 
						|
		  2>/dev/null > $tmp
 | 
						|
	    $dd if=$tmp of=$ed.c 2>/dev/null
 | 
						|
	    return 0
 | 
						|
	  else
 | 
						|
	    subtract n $n 1
 | 
						|
	    ( $dd if=$tmp.4; $dd if=$tmp.5 bs=1 count=1 ) > $tmp 2>/dev/null
 | 
						|
	    $dd if=$tmp of=$tmp.4 2>/dev/null
 | 
						|
	    # and remove it from $tmp.5
 | 
						|
	    $dd if=$tmp.5 of=$tmp bs=1 skip=1 2>/dev/null
 | 
						|
	    $dd if=$tmp of=$tmp.5 2>/dev/null
 | 
						|
	  fi
 | 
						|
	  ;;
 | 
						|
      ?*) # add one more byte...
 | 
						|
	  ( $dd if=$tmp.4; $dd if=$tmp.5 bs=1 count=1 ) > $tmp 2>/dev/null
 | 
						|
	  $dd if=$tmp of=$tmp.4 2>/dev/null
 | 
						|
	  # and remove it from $tmp.5
 | 
						|
	  $dd if=$tmp.5 of=$tmp bs=1 skip=1 2>/dev/null
 | 
						|
	  $dd if=$tmp of=$tmp.5 2>/dev/null
 | 
						|
	  ;;
 | 
						|
      *)  # not found
 | 
						|
	  return 1
 | 
						|
	  ;;
 | 
						|
    esac
 | 
						|
  done
 | 
						|
}
 | 
						|
 | 
						|
# skip to next line
 | 
						|
next () {
 | 
						|
  add l $lineno 1
 | 
						|
  ( $dd if=$ed.a; $dd if=$ed.c ) 2>/dev/null > $tmp.3
 | 
						|
  $dd if=$ed.b of=$tmp.4 2>/dev/null
 | 
						|
  open $tmp.4
 | 
						|
  $dd if=$tmp.3 of=$ed.a 2>/dev/null
 | 
						|
  lineno=$l
 | 
						|
}
 | 
						|
 | 
						|
# delete current line
 | 
						|
delete () {
 | 
						|
  l=$lineno
 | 
						|
  $dd if=$ed.a 2>/dev/null > $tmp.1
 | 
						|
  $dd if=$ed.b of=$tmp.2 2>/dev/null
 | 
						|
  open $tmp.2
 | 
						|
  $dd if=$tmp.1 of=$ed.a 2>/dev/null
 | 
						|
  lineno=$l
 | 
						|
}
 | 
						|
 | 
						|
# insert before current line (without changing current)
 | 
						|
insert () {
 | 
						|
  ( $dd if=$ed.a; Echo "$@" ) 2>/dev/null > $tmp.1
 | 
						|
  $dd if=$tmp.1 of=$ed.a 2>/dev/null
 | 
						|
  add lineno $lineno 1
 | 
						|
}
 | 
						|
 | 
						|
# previous line
 | 
						|
prev () {
 | 
						|
  case "$lineno" in
 | 
						|
    1) ;;
 | 
						|
    *) subtract lineno $lineno 1
 | 
						|
       # read last line of $ed.a
 | 
						|
       IFS='+'
 | 
						|
       set `$dd if=$ed.a of=/dev/null bs=1 2>&1`
 | 
						|
       IFS="$saveIFS"
 | 
						|
       size=$1
 | 
						|
       # empty?
 | 
						|
       case "$size" in
 | 
						|
	 0) return ;;
 | 
						|
       esac
 | 
						|
       subtract size $size 1
 | 
						|
       # skip final newline
 | 
						|
       case "$size" in
 | 
						|
	 0) ;;
 | 
						|
	 *) subtract size1 $size 1
 | 
						|
	    case "`$dd if=$ed.a bs=1 skip=$size count=1 2>/dev/null`" in
 | 
						|
	      ?*) ;;
 | 
						|
	      *) size=$size1 ;;
 | 
						|
	    esac
 | 
						|
	    ;;
 | 
						|
       esac
 | 
						|
       go=true
 | 
						|
       while $go
 | 
						|
       do
 | 
						|
	 case "$size" in
 | 
						|
	   0) go=false ;;
 | 
						|
	   *) case "`$dd if=$ed.a bs=1 skip=$size count=1 2>/dev/null`" in
 | 
						|
	        ?*)  go=true; subtract size $size 1 ;;
 | 
						|
	        *)   go=false; add size $size 1 ;;
 | 
						|
	      esac
 | 
						|
	      ;;
 | 
						|
	 esac
 | 
						|
       done
 | 
						|
       # now $size is the size of the first n-1 lines
 | 
						|
       # add $ed.c to $ed.b
 | 
						|
       ( $dd if=$ed.c; $dd if=$ed.b ) 2>/dev/null > $tmp.5
 | 
						|
       $dd if=$tmp.5 of=$ed.b 2>/dev/null
 | 
						|
       # move line to ed.c
 | 
						|
       case "$size" in
 | 
						|
	 0) $dd if=$ed.a of=$ed.c 2>/dev/null
 | 
						|
	    $dd if=/dev/null of=$tmp.5 2>/dev/null
 | 
						|
	    ;;
 | 
						|
	 *) $dd if=$ed.a of=$ed.c bs=1 skip=$size 2>/dev/null
 | 
						|
	    $dd if=$ed.a of=$tmp.5 bs=1 count=$size 2>/dev/null
 | 
						|
	    ;;
 | 
						|
       esac
 | 
						|
       # move rest to ed.a
 | 
						|
       $dd if=$tmp.5 of=$ed.a 2>/dev/null
 | 
						|
    ;;
 | 
						|
  esac
 | 
						|
}
 | 
						|
 | 
						|
# goes to a given line
 | 
						|
goto () {
 | 
						|
  rl="$1"
 | 
						|
  compare bla "$rl" $lineno
 | 
						|
  case "$bla" in
 | 
						|
    eq) return
 | 
						|
	;;
 | 
						|
    gt) while gt "$rl" $lineno
 | 
						|
	do
 | 
						|
	  next
 | 
						|
	done
 | 
						|
	;;
 | 
						|
    lt) while lt "$rl" $lineno
 | 
						|
	do
 | 
						|
	  prev
 | 
						|
	done
 | 
						|
	;;
 | 
						|
  esac
 | 
						|
}
 | 
						|
 | 
						|
lineout () {
 | 
						|
  Echo -n "$lineno: "
 | 
						|
  $dd if=$ed.c 2>/dev/null
 | 
						|
}
 | 
						|
 | 
						|
state=closed
 | 
						|
name=
 | 
						|
autoprint=true
 | 
						|
 | 
						|
while true
 | 
						|
do
 | 
						|
  Echo -n '> '
 | 
						|
  read cmd arg
 | 
						|
  case "$cmd:$state" in
 | 
						|
    open:open) Echo "There is a file open already" ;;
 | 
						|
    open:*) if open "$arg"
 | 
						|
	    then state=open; name="$arg"; $autoprint
 | 
						|
	    else Echo "Cannot open $arg"
 | 
						|
	    fi
 | 
						|
	    ;;
 | 
						|
    new:open) Echo "There is a file open already" ;;
 | 
						|
    new:*)  open "$arg"
 | 
						|
	    state=open
 | 
						|
	    name="$arg"
 | 
						|
	    $autoprint
 | 
						|
	    ;;
 | 
						|
    close:changed) Echo "Use 'discard' or 'save'" ;;
 | 
						|
    close:closed) Echo "Closed already" ;;
 | 
						|
    close:*) state=closed ;;
 | 
						|
    save:closed) Echo "There isn't a file to save" ;;
 | 
						|
    save:*) case "$arg" in
 | 
						|
	      ?*) save "$arg" ;;
 | 
						|
	      *) save "$name" ;;
 | 
						|
	    esac
 | 
						|
	    state=open
 | 
						|
	    ;;
 | 
						|
    discard:changed) Echo "Your problem!"; state=closed ;;
 | 
						|
    discard:*) state=closed ;;
 | 
						|
    print:closed) Echo "No current file" ;;
 | 
						|
    print:*) lineout ;;
 | 
						|
    goto:closed) Echo "No current file" ;;
 | 
						|
    goto:*) goto "$arg"; $autoprint ;;
 | 
						|
    next:closed) Echo "No current file" ;;
 | 
						|
    next:*) next; $autoprint ;;
 | 
						|
    prev:closed) Echo "No current file" ;;
 | 
						|
    prev:*) prev; $autoprint ;;
 | 
						|
    name:closed) Echo "No current file" ;;
 | 
						|
    name:*) name="$arg" ;;
 | 
						|
    replace:closed) Echo "No current file" ;;
 | 
						|
    replace:*) if rstring 1 $arg
 | 
						|
	       then state=changed; $autoprint
 | 
						|
	       else Echo "Not found"
 | 
						|
	       fi
 | 
						|
	       ;;
 | 
						|
    nreplace:closed) Echo "No current file" ;;
 | 
						|
    nreplace:*) if rstring $arg
 | 
						|
		then state=changed; $autoprint
 | 
						|
		else Echo "Not found"
 | 
						|
		fi
 | 
						|
		;;
 | 
						|
    delete:closed) Echo "No current file" ;;
 | 
						|
    delete:*) delete; state=changed; $autoprint ;;
 | 
						|
    insert:closed) Echo "No current file" ;;
 | 
						|
    insert:*) insert "$arg"; prev; state=changed; $autoprint ;;
 | 
						|
    quit:changed) Echo "Use 'save' or 'discard'" ;;
 | 
						|
    quit:*) Echo "bye"; exit;;
 | 
						|
    autoprint:*) autoprint="lineout" ;;
 | 
						|
    noprint:*) autoprint="" ;;
 | 
						|
    :*) ;;
 | 
						|
    *) Echo "Command not understood" ;;
 | 
						|
  esac
 | 
						|
done
 | 
						|
 |