[dm-devel] dmraid autoconf/config.guess autoconf/config.s ...

heinzm at sourceware.org heinzm at sourceware.org
Fri Feb 22 16:57:45 UTC 2008


CVSROOT:	/cvs/dm
Module name:	dmraid
Changes by:	heinzm at sourceware.org	2008-02-22 16:57:37

Added files:
	autoconf       : config.guess config.sub install-sh 
	doc            : dmraid_design.txt 
	include        : Makefile.in 
	include/dmraid : display.h dmraid.h format.h lib_context.h 
	                 list.h locking.h metadata.h misc.h 
	lib            : .export.sym Makefile.in internal.h version.h 
	lib/activate   : activate.c activate.h devmapper.c devmapper.h 
	lib/datastruct : byteorder.h 
	lib/device     : ata.c ata.h dev-io.h scan.c scsi.c scsi.h 
	lib/display    : display.c 
	lib/format     : README format.c ondisk.h register.h 
	lib/format/ataraid: README asr.c asr.h hpt37x.c hpt37x.h 
	                    hpt45x.c hpt45x.h isw.c isw.h jm.c jm.h 
	                    lsi.c lsi.h nv.c nv.h pdc.c pdc.h sil.c 
	                    sil.h via.c via.h 
	lib/format/partition: dos.c dos.h 
	lib/format/template: README template.c template.h 
	lib/locking    : locking.c 
	lib/log        : log.c log.h 
	lib/metadata   : metadata.c 
	lib/misc       : file.c init.c lib_context.c misc.c workaround.c 
	lib/mm         : dbg_malloc.c dbg_malloc.h 
	man            : Makefile.in dmraid.8 
	tools          : Makefile.in VERSION commands.c commands.h 
	                 dmraid.c toollib.c toollib.h version.h.in 

Log message:
	initial 1.0.0.rc11 checkin (missing files)

Patches:
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/autoconf/config.guess.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/autoconf/config.sub.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/autoconf/install-sh.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/doc/dmraid_design.txt.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/Makefile.in.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/display.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/dmraid.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/format.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/lib_context.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/list.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/locking.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/metadata.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/include/dmraid/misc.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/.export.sym.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/Makefile.in.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/internal.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/version.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/activate/activate.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/activate/activate.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/activate/devmapper.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/activate/devmapper.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/datastruct/byteorder.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/device/ata.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/device/ata.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/device/dev-io.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/device/scan.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/device/scsi.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/device/scsi.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/display/display.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/README.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/format.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ondisk.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/register.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/README.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/asr.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/asr.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/hpt37x.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/hpt37x.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/hpt45x.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/hpt45x.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/isw.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/isw.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/jm.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/jm.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/lsi.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/lsi.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/nv.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/nv.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/pdc.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/pdc.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/sil.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/sil.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/via.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/ataraid/via.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/partition/dos.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/partition/dos.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/template/README.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/template/template.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/format/template/template.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/locking/locking.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/log/log.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/log/log.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/metadata/metadata.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/misc/file.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/misc/init.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/misc/lib_context.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/misc/misc.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/misc/workaround.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/mm/dbg_malloc.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/lib/mm/dbg_malloc.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/man/Makefile.in.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/man/dmraid.8.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/Makefile.in.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/VERSION.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/commands.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/commands.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/dmraid.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/toollib.c.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/toollib.h.diff?cvsroot=dm&r1=NONE&r2=1.1
http://sourceware.org/cgi-bin/cvsweb.cgi/dmraid/tools/version.h.in.diff?cvsroot=dm&r1=NONE&r2=1.1

/cvs/dm/dmraid/autoconf/config.guess,v  -->  standard output
revision 1.1
--- dmraid/autoconf/config.guess
+++ -	2008-02-22 16:57:37.811092000 +0000
@@ -0,0 +1,1404 @@
+#! /bin/sh
+# Attempt to guess a canonical system name.
+#   Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999,
+#   2000, 2001, 2002, 2003 Free Software Foundation, Inc.
+
+timestamp='2003-06-17'
+
+# This file is free software; you can redistribute it and/or modify it
+# under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful, but
+# WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+# General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+#
+# As a special exception to the GNU General Public License, if you
+# distribute this file as part of a program that contains a
+# configuration script generated by Autoconf, you may include it under
+# the same distribution terms that you use for the rest of that program.
+
+# Originally written by Per Bothner <per at bothner.com>.
+# Please send patches to <config-patches at gnu.org>.  Submit a context
+# diff and a properly formatted ChangeLog entry.
+#
+# This script attempts to guess a canonical system name similar to
+# config.sub.  If it succeeds, it prints the system name on stdout, and
+# exits with 0.  Otherwise, it exits with 1.
+#
+# The plan is that this can be called by configure scripts if you
+# don't specify an explicit build system type.
+
+me=`echo "$0" | sed -e 's,.*/,,'`
+
+usage="\
+Usage: $0 [OPTION]
+
+Output the configuration name of the system \`$me' is run on.
+
+Operation modes:
+  -h, --help         print this help, then exit
+  -t, --time-stamp   print date of last modification, then exit
+  -v, --version      print version number, then exit
+
+Report bugs and patches to <config-patches at gnu.org>."
+
+version="\
+GNU config.guess ($timestamp)
+
+Originally written by Per Bothner.
+Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001
+Free Software Foundation, Inc.
+
+This is free software; see the source for copying conditions.  There is NO
+warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE."
+
+help="
+Try \`$me --help' for more information."
+
+# Parse command line
+while test $# -gt 0 ; do
+  case $1 in
+    --time-stamp | --time* | -t )
+       echo "$timestamp" ; exit 0 ;;
+    --version | -v )
+       echo "$version" ; exit 0 ;;
+    --help | --h* | -h )
+       echo "$usage"; exit 0 ;;
+    -- )     # Stop option processing
+       shift; break ;;
+    - )	# Use stdin as input.
+       break ;;
+    -* )
+       echo "$me: invalid option $1$help" >&2
+       exit 1 ;;
+    * )
+       break ;;
+  esac
+done
+
+if test $# != 0; then
+  echo "$me: too many arguments$help" >&2
+  exit 1
+fi
+
+trap 'exit 1' 1 2 15
+
+# CC_FOR_BUILD -- compiler used by this script. Note that the use of a
+# compiler to aid in system detection is discouraged as it requires
+# temporary files to be created and, as you can see below, it is a
+# headache to deal with in a portable fashion.
+
+# Historically, `CC_FOR_BUILD' used to be named `HOST_CC'. We still
+# use `HOST_CC' if defined, but it is deprecated.
+
+# Portable tmp directory creation inspired by the Autoconf team.
+
+set_cc_for_build='
+trap "exitcode=\$?; (rm -f \$tmpfiles 2>/dev/null; rmdir \$tmp 2>/dev/null) && exit \$exitcode" 0 ;
+trap "rm -f \$tmpfiles 2>/dev/null; rmdir \$tmp 2>/dev/null; exit 1" 1 2 13 15 ;
+: ${TMPDIR=/tmp} ;
+ { tmp=`(umask 077 && mktemp -d -q "$TMPDIR/cgXXXXXX") 2>/dev/null` && test -n "$tmp" && test -d "$tmp" ; } ||
+ { test -n "$RANDOM" && tmp=$TMPDIR/cg$$-$RANDOM && (umask 077 && mkdir $tmp) ; } ||
+ { tmp=$TMPDIR/cg-$$ && (umask 077 && mkdir $tmp) && echo "Warning: creating insecure temp directory" >&2 ; } ||
+ { echo "$me: cannot create a temporary directory in $TMPDIR" >&2 ; exit 1 ; } ;
+dummy=$tmp/dummy ;
+tmpfiles="$dummy.c $dummy.o $dummy.rel $dummy" ;
+case $CC_FOR_BUILD,$HOST_CC,$CC in
+ ,,)    echo "int x;" > $dummy.c ;
+	for c in cc gcc c89 c99 ; do
+	  if ($c -c -o $dummy.o $dummy.c) >/dev/null 2>&1 ; then
+	     CC_FOR_BUILD="$c"; break ;
+	  fi ;
+	done ;
+	if test x"$CC_FOR_BUILD" = x ; then
+	  CC_FOR_BUILD=no_compiler_found ;
+	fi
+	;;
+ ,,*)   CC_FOR_BUILD=$CC ;;
+ ,*,*)  CC_FOR_BUILD=$HOST_CC ;;
+esac ;'
+
+# This is needed to find uname on a Pyramid OSx when run in the BSD universe.
+# (ghazi at noc.rutgers.edu 1994-08-24)
+if (test -f /.attbin/uname) >/dev/null 2>&1 ; then
+	PATH=$PATH:/.attbin ; export PATH
+fi
+
+UNAME_MACHINE=`(uname -m) 2>/dev/null` || UNAME_MACHINE=unknown
+UNAME_RELEASE=`(uname -r) 2>/dev/null` || UNAME_RELEASE=unknown
+UNAME_SYSTEM=`(uname -s) 2>/dev/null`  || UNAME_SYSTEM=unknown
+UNAME_VERSION=`(uname -v) 2>/dev/null` || UNAME_VERSION=unknown
+
+# Note: order is significant - the case branches are not exclusive.
+
+case "${UNAME_MACHINE}:${UNAME_SYSTEM}:${UNAME_RELEASE}:${UNAME_VERSION}" in
+    *:NetBSD:*:*)
+	# NetBSD (nbsd) targets should (where applicable) match one or
+	# more of the tupples: *-*-netbsdelf*, *-*-netbsdaout*,
+	# *-*-netbsdecoff* and *-*-netbsd*.  For targets that recently
+	# switched to ELF, *-*-netbsd* would select the old
+	# object file format.  This provides both forward
+	# compatibility and a consistent mechanism for selecting the
+	# object file format.
+	#
+	# Note: NetBSD doesn't particularly care about the vendor
+	# portion of the name.  We always set it to "unknown".
+	sysctl="sysctl -n hw.machine_arch"
+	UNAME_MACHINE_ARCH=`(/sbin/$sysctl 2>/dev/null || \
+	    /usr/sbin/$sysctl 2>/dev/null || echo unknown)`
+	case "${UNAME_MACHINE_ARCH}" in
+	    armeb) machine=armeb-unknown ;;
+	    arm*) machine=arm-unknown ;;
+	    sh3el) machine=shl-unknown ;;
+	    sh3eb) machine=sh-unknown ;;
+	    *) machine=${UNAME_MACHINE_ARCH}-unknown ;;
+	esac
+	# The Operating System including object format, if it has switched
+	# to ELF recently, or will in the future.
+	case "${UNAME_MACHINE_ARCH}" in
+	    arm*|i386|m68k|ns32k|sh3*|sparc|vax)
+		eval $set_cc_for_build
+		if echo __ELF__ | $CC_FOR_BUILD -E - 2>/dev/null \
+			| grep __ELF__ >/dev/null
+		then
+		    # Once all utilities can be ECOFF (netbsdecoff) or a.out (netbsdaout).
+		    # Return netbsd for either.  FIX?
+		    os=netbsd
+		else
+		    os=netbsdelf
+		fi
+		;;
+	    *)
+	        os=netbsd
+		;;
+	esac
+	# The OS release
+	# Debian GNU/NetBSD machines have a different userland, and
+	# thus, need a distinct triplet. However, they do not need
+	# kernel version information, so it can be replaced with a
+	# suitable tag, in the style of linux-gnu.
+	case "${UNAME_VERSION}" in
+	    Debian*)
+		release='-gnu'
+		;;
+	    *)
+		release=`echo ${UNAME_RELEASE}|sed -e 's/[-_].*/\./'`
+		;;
+	esac
+	# Since CPU_TYPE-MANUFACTURER-KERNEL-OPERATING_SYSTEM:
+	# contains redundant information, the shorter form:
+	# CPU_TYPE-MANUFACTURER-OPERATING_SYSTEM is used.
+	echo "${machine}-${os}${release}"
+	exit 0 ;;
+    amiga:OpenBSD:*:*)
+	echo m68k-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    arc:OpenBSD:*:*)
+	echo mipsel-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    hp300:OpenBSD:*:*)
+	echo m68k-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    mac68k:OpenBSD:*:*)
+	echo m68k-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    macppc:OpenBSD:*:*)
+	echo powerpc-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    mvme68k:OpenBSD:*:*)
+	echo m68k-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    mvme88k:OpenBSD:*:*)
+	echo m88k-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    mvmeppc:OpenBSD:*:*)
+	echo powerpc-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    pmax:OpenBSD:*:*)
+	echo mipsel-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    sgi:OpenBSD:*:*)
+	echo mipseb-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    sun3:OpenBSD:*:*)
+	echo m68k-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    wgrisc:OpenBSD:*:*)
+	echo mipsel-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    *:OpenBSD:*:*)
+	echo ${UNAME_MACHINE}-unknown-openbsd${UNAME_RELEASE}
+	exit 0 ;;
+    alpha:OSF1:*:*)
+	if test $UNAME_RELEASE = "V4.0"; then
+		UNAME_RELEASE=`/usr/sbin/sizer -v | awk '{print $3}'`
+	fi
+	# According to Compaq, /usr/sbin/psrinfo has been available on
+	# OSF/1 and Tru64 systems produced since 1995.  I hope that
+	# covers most systems running today.  This code pipes the CPU
+	# types through head -n 1, so we only detect the type of CPU 0.
+	ALPHA_CPU_TYPE=`/usr/sbin/psrinfo -v | sed -n -e 's/^  The alpha \(.*\) processor.*$/\1/p' | head -n 1`
+	case "$ALPHA_CPU_TYPE" in
+	    "EV4 (21064)")
+		UNAME_MACHINE="alpha" ;;
+	    "EV4.5 (21064)")
+		UNAME_MACHINE="alpha" ;;
+	    "LCA4 (21066/21068)")
+		UNAME_MACHINE="alpha" ;;
+	    "EV5 (21164)")
+		UNAME_MACHINE="alphaev5" ;;
+	    "EV5.6 (21164A)")
+		UNAME_MACHINE="alphaev56" ;;
+	    "EV5.6 (21164PC)")
+		UNAME_MACHINE="alphapca56" ;;
+	    "EV5.7 (21164PC)")
+		UNAME_MACHINE="alphapca57" ;;
+	    "EV6 (21264)")
+		UNAME_MACHINE="alphaev6" ;;
+	    "EV6.7 (21264A)")
+		UNAME_MACHINE="alphaev67" ;;
+	    "EV6.8CB (21264C)")
+		UNAME_MACHINE="alphaev68" ;;
+	    "EV6.8AL (21264B)")
+		UNAME_MACHINE="alphaev68" ;;
+	    "EV6.8CX (21264D)")
+		UNAME_MACHINE="alphaev68" ;;
+	    "EV6.9A (21264/EV69A)")
+		UNAME_MACHINE="alphaev69" ;;
+	    "EV7 (21364)")
+		UNAME_MACHINE="alphaev7" ;;
+	    "EV7.9 (21364A)")
+		UNAME_MACHINE="alphaev79" ;;
+	esac
+	# A Vn.n version is a released version.
+	# A Tn.n version is a released field test version.
+	# A Xn.n version is an unreleased experimental baselevel.
+	# 1.2 uses "1.2" for uname -r.
+	echo ${UNAME_MACHINE}-dec-osf`echo ${UNAME_RELEASE} | sed -e 's/^[VTX]//' | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'`
+	exit 0 ;;
+    Alpha*:OpenVMS:*:*)
+	echo alpha-hp-vms
+	exit 0 ;;
+    Alpha\ *:Windows_NT*:*)
+	# How do we know it's Interix rather than the generic POSIX subsystem?
+	# Should we change UNAME_MACHINE based on the output of uname instead
+	# of the specific Alpha model?
+	echo alpha-pc-interix
+	exit 0 ;;
+    21064:Windows_NT:50:3)
+	echo alpha-dec-winnt3.5
+	exit 0 ;;
+    Amiga*:UNIX_System_V:4.0:*)
+	echo m68k-unknown-sysv4
+	exit 0;;
+    *:[Aa]miga[Oo][Ss]:*:*)
+	echo ${UNAME_MACHINE}-unknown-amigaos
+	exit 0 ;;
+    *:[Mm]orph[Oo][Ss]:*:*)
+	echo ${UNAME_MACHINE}-unknown-morphos
+	exit 0 ;;
+    *:OS/390:*:*)
+	echo i370-ibm-openedition
+	exit 0 ;;
+    arm:RISC*:1.[012]*:*|arm:riscix:1.[012]*:*)
+	echo arm-acorn-riscix${UNAME_RELEASE}
+	exit 0;;
+    SR2?01:HI-UX/MPP:*:* | SR8000:HI-UX/MPP:*:*)
+	echo hppa1.1-hitachi-hiuxmpp
+	exit 0;;
+    Pyramid*:OSx*:*:* | MIS*:OSx*:*:* | MIS*:SMP_DC-OSx*:*:*)
+	# akee at wpdis03.wpafb.af.mil (Earle F. Ake) contributed MIS and NILE.
+	if test "`(/bin/universe) 2>/dev/null`" = att ; then
+		echo pyramid-pyramid-sysv3
+	else
+		echo pyramid-pyramid-bsd
+	fi
+	exit 0 ;;
+    NILE*:*:*:dcosx)
+	echo pyramid-pyramid-svr4
+	exit 0 ;;
+    DRS?6000:unix:4.0:6*)
+	echo sparc-icl-nx6
+	exit 0 ;;
+    DRS?6000:UNIX_SV:4.2*:7*)
+	case `/usr/bin/uname -p` in
+	    sparc) echo sparc-icl-nx7 && exit 0 ;;
+	esac ;;
+    sun4H:SunOS:5.*:*)
+	echo sparc-hal-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'`
+	exit 0 ;;
+    sun4*:SunOS:5.*:* | tadpole*:SunOS:5.*:*)
+	echo sparc-sun-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'`
+	exit 0 ;;
+    i86pc:SunOS:5.*:*)
+	echo i386-pc-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'`
+	exit 0 ;;
+    sun4*:SunOS:6*:*)
+	# According to config.sub, this is the proper way to canonicalize
+	# SunOS6.  Hard to guess exactly what SunOS6 will be like, but
+	# it's likely to be more like Solaris than SunOS4.
+	echo sparc-sun-solaris3`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'`
+	exit 0 ;;
+    sun4*:SunOS:*:*)
+	case "`/usr/bin/arch -k`" in
+	    Series*|S4*)
+		UNAME_RELEASE=`uname -v`
+		;;
+	esac
+	# Japanese Language versions have a version number like `4.1.3-JL'.
+	echo sparc-sun-sunos`echo ${UNAME_RELEASE}|sed -e 's/-/_/'`
+	exit 0 ;;
+    sun3*:SunOS:*:*)
+	echo m68k-sun-sunos${UNAME_RELEASE}
+	exit 0 ;;
+    sun*:*:4.2BSD:*)
+	UNAME_RELEASE=`(sed 1q /etc/motd | awk '{print substr($5,1,3)}') 2>/dev/null`
+	test "x${UNAME_RELEASE}" = "x" && UNAME_RELEASE=3
+	case "`/bin/arch`" in
+	    sun3)
+		echo m68k-sun-sunos${UNAME_RELEASE}
+		;;
+	    sun4)
+		echo sparc-sun-sunos${UNAME_RELEASE}
+		;;
+	esac
+	exit 0 ;;
+    aushp:SunOS:*:*)
+	echo sparc-auspex-sunos${UNAME_RELEASE}
+	exit 0 ;;
+    # The situation for MiNT is a little confusing.  The machine name
+    # can be virtually everything (everything which is not
+    # "atarist" or "atariste" at least should have a processor
+    # > m68000).  The system name ranges from "MiNT" over "FreeMiNT"
+    # to the lowercase version "mint" (or "freemint").  Finally
+    # the system name "TOS" denotes a system which is actually not
+    # MiNT.  But MiNT is downward compatible to TOS, so this should
+    # be no problem.
+    atarist[e]:*MiNT:*:* | atarist[e]:*mint:*:* | atarist[e]:*TOS:*:*)
+        echo m68k-atari-mint${UNAME_RELEASE}
+	exit 0 ;;
+    atari*:*MiNT:*:* | atari*:*mint:*:* | atarist[e]:*TOS:*:*)
+	echo m68k-atari-mint${UNAME_RELEASE}
+        exit 0 ;;
+    *falcon*:*MiNT:*:* | *falcon*:*mint:*:* | *falcon*:*TOS:*:*)
+        echo m68k-atari-mint${UNAME_RELEASE}
+	exit 0 ;;
+    milan*:*MiNT:*:* | milan*:*mint:*:* | *milan*:*TOS:*:*)
+        echo m68k-milan-mint${UNAME_RELEASE}
+        exit 0 ;;
+    hades*:*MiNT:*:* | hades*:*mint:*:* | *hades*:*TOS:*:*)
+        echo m68k-hades-mint${UNAME_RELEASE}
+        exit 0 ;;
+    *:*MiNT:*:* | *:*mint:*:* | *:*TOS:*:*)
+        echo m68k-unknown-mint${UNAME_RELEASE}
+        exit 0 ;;
+    powerpc:machten:*:*)
+	echo powerpc-apple-machten${UNAME_RELEASE}
+	exit 0 ;;
+    RISC*:Mach:*:*)
+	echo mips-dec-mach_bsd4.3
+	exit 0 ;;
+    RISC*:ULTRIX:*:*)
+	echo mips-dec-ultrix${UNAME_RELEASE}
+	exit 0 ;;
+    VAX*:ULTRIX*:*:*)
+	echo vax-dec-ultrix${UNAME_RELEASE}
+	exit 0 ;;
+    2020:CLIX:*:* | 2430:CLIX:*:*)
+	echo clipper-intergraph-clix${UNAME_RELEASE}
+	exit 0 ;;
+    mips:*:*:UMIPS | mips:*:*:RISCos)
+	eval $set_cc_for_build
+	sed 's/^	//' << EOF >$dummy.c
+#ifdef __cplusplus
+#include <stdio.h>  /* for printf() prototype */
+	int main (int argc, char *argv[]) {
+#else
+	int main (argc, argv) int argc; char *argv[]; {
+#endif
+	#if defined (host_mips) && defined (MIPSEB)
+	#if defined (SYSTYPE_SYSV)
+	  printf ("mips-mips-riscos%ssysv\n", argv[1]); exit (0);
+	#endif
+	#if defined (SYSTYPE_SVR4)
+	  printf ("mips-mips-riscos%ssvr4\n", argv[1]); exit (0);
+	#endif
+	#if defined (SYSTYPE_BSD43) || defined(SYSTYPE_BSD)
+	  printf ("mips-mips-riscos%sbsd\n", argv[1]); exit (0);
+	#endif
+	#endif
+	  exit (-1);
+	}
+EOF
+	$CC_FOR_BUILD -o $dummy $dummy.c \
+	  && $dummy `echo "${UNAME_RELEASE}" | sed -n 's/\([0-9]*\).*/\1/p'` \
+	  && exit 0
+	echo mips-mips-riscos${UNAME_RELEASE}
+	exit 0 ;;
+    Motorola:PowerMAX_OS:*:*)
+	echo powerpc-motorola-powermax
+	exit 0 ;;
+    Motorola:*:4.3:PL8-*)
+	echo powerpc-harris-powermax
+	exit 0 ;;
+    Night_Hawk:*:*:PowerMAX_OS | Synergy:PowerMAX_OS:*:*)
+	echo powerpc-harris-powermax
+	exit 0 ;;
+    Night_Hawk:Power_UNIX:*:*)
+	echo powerpc-harris-powerunix
+	exit 0 ;;
+    m88k:CX/UX:7*:*)
+	echo m88k-harris-cxux7
+	exit 0 ;;
+    m88k:*:4*:R4*)
+	echo m88k-motorola-sysv4
+	exit 0 ;;
+    m88k:*:3*:R3*)
+	echo m88k-motorola-sysv3
+	exit 0 ;;
+    AViiON:dgux:*:*)
+        # DG/UX returns AViiON for all architectures
+        UNAME_PROCESSOR=`/usr/bin/uname -p`
+	if [ $UNAME_PROCESSOR = mc88100 ] || [ $UNAME_PROCESSOR = mc88110 ]
+	then
+	    if [ ${TARGET_BINARY_INTERFACE}x = m88kdguxelfx ] || \
+	       [ ${TARGET_BINARY_INTERFACE}x = x ]
+	    then
+		echo m88k-dg-dgux${UNAME_RELEASE}
+	    else
+		echo m88k-dg-dguxbcs${UNAME_RELEASE}
+	    fi
+	else
+	    echo i586-dg-dgux${UNAME_RELEASE}
+	fi
+ 	exit 0 ;;
+    M88*:DolphinOS:*:*)	# DolphinOS (SVR3)
+	echo m88k-dolphin-sysv3
+	exit 0 ;;
+    M88*:*:R3*:*)
+	# Delta 88k system running SVR3
+	echo m88k-motorola-sysv3
+	exit 0 ;;
+    XD88*:*:*:*) # Tektronix XD88 system running UTekV (SVR3)
+	echo m88k-tektronix-sysv3
+	exit 0 ;;
+    Tek43[0-9][0-9]:UTek:*:*) # Tektronix 4300 system running UTek (BSD)
+	echo m68k-tektronix-bsd
+	exit 0 ;;
+    *:IRIX*:*:*)
+	echo mips-sgi-irix`echo ${UNAME_RELEASE}|sed -e 's/-/_/g'`
+	exit 0 ;;
+    ????????:AIX?:[12].1:2)   # AIX 2.2.1 or AIX 2.1.1 is RT/PC AIX.
+	echo romp-ibm-aix      # uname -m gives an 8 hex-code CPU id
+	exit 0 ;;              # Note that: echo "'`uname -s`'" gives 'AIX '
+    i*86:AIX:*:*)
+	echo i386-ibm-aix
+	exit 0 ;;
+    ia64:AIX:*:*)
+	if [ -x /usr/bin/oslevel ] ; then
+		IBM_REV=`/usr/bin/oslevel`
+	else
+		IBM_REV=${UNAME_VERSION}.${UNAME_RELEASE}
+	fi
+	echo ${UNAME_MACHINE}-ibm-aix${IBM_REV}
+	exit 0 ;;
+    *:AIX:2:3)
+	if grep bos325 /usr/include/stdio.h >/dev/null 2>&1; then
+		eval $set_cc_for_build
+		sed 's/^		//' << EOF >$dummy.c
+		#include <sys/systemcfg.h>
+
+		main()
+			{
+			if (!__power_pc())
+				exit(1);
+			puts("powerpc-ibm-aix3.2.5");
+			exit(0);
+			}
+EOF
+		$CC_FOR_BUILD -o $dummy $dummy.c && $dummy && exit 0
+		echo rs6000-ibm-aix3.2.5
+	elif grep bos324 /usr/include/stdio.h >/dev/null 2>&1; then
+		echo rs6000-ibm-aix3.2.4
+	else
+		echo rs6000-ibm-aix3.2
+	fi
+	exit 0 ;;
+    *:AIX:*:[45])
+	IBM_CPU_ID=`/usr/sbin/lsdev -C -c processor -S available | sed 1q | awk '{ print $1 }'`
+	if /usr/sbin/lsattr -El ${IBM_CPU_ID} | grep ' POWER' >/dev/null 2>&1; then
+		IBM_ARCH=rs6000
+	else
+		IBM_ARCH=powerpc
+	fi
+	if [ -x /usr/bin/oslevel ] ; then
+		IBM_REV=`/usr/bin/oslevel`
+	else
+		IBM_REV=${UNAME_VERSION}.${UNAME_RELEASE}
+	fi
+	echo ${IBM_ARCH}-ibm-aix${IBM_REV}
+	exit 0 ;;
+    *:AIX:*:*)
+	echo rs6000-ibm-aix
+	exit 0 ;;
+    ibmrt:4.4BSD:*|romp-ibm:BSD:*)
+	echo romp-ibm-bsd4.4
+	exit 0 ;;
+    ibmrt:*BSD:*|romp-ibm:BSD:*)            # covers RT/PC BSD and
+	echo romp-ibm-bsd${UNAME_RELEASE}   # 4.3 with uname added to
+	exit 0 ;;                           # report: romp-ibm BSD 4.3
+    *:BOSX:*:*)
+	echo rs6000-bull-bosx
+	exit 0 ;;
+    DPX/2?00:B.O.S.:*:*)
+	echo m68k-bull-sysv3
+	exit 0 ;;
+    9000/[34]??:4.3bsd:1.*:*)
+	echo m68k-hp-bsd
+	exit 0 ;;
+    hp300:4.4BSD:*:* | 9000/[34]??:4.3bsd:2.*:*)
+	echo m68k-hp-bsd4.4
+	exit 0 ;;
+    9000/[34678]??:HP-UX:*:*)
+	HPUX_REV=`echo ${UNAME_RELEASE}|sed -e 's/[^.]*.[0B]*//'`
+	case "${UNAME_MACHINE}" in
+	    9000/31? )            HP_ARCH=m68000 ;;
+	    9000/[34]?? )         HP_ARCH=m68k ;;
+	    9000/[678][0-9][0-9])
+		if [ -x /usr/bin/getconf ]; then
+		    sc_cpu_version=`/usr/bin/getconf SC_CPU_VERSION 2>/dev/null`
+                    sc_kernel_bits=`/usr/bin/getconf SC_KERNEL_BITS 2>/dev/null`
+                    case "${sc_cpu_version}" in
+                      523) HP_ARCH="hppa1.0" ;; # CPU_PA_RISC1_0
+                      528) HP_ARCH="hppa1.1" ;; # CPU_PA_RISC1_1
+                      532)                      # CPU_PA_RISC2_0
+                        case "${sc_kernel_bits}" in
+                          32) HP_ARCH="hppa2.0n" ;;
+                          64) HP_ARCH="hppa2.0w" ;;
+			  '') HP_ARCH="hppa2.0" ;;   # HP-UX 10.20
+                        esac ;;
+                    esac
+		fi
+		if [ "${HP_ARCH}" = "" ]; then
+		    eval $set_cc_for_build
+		    sed 's/^              //' << EOF >$dummy.c
+
+              #define _HPUX_SOURCE
+              #include <stdlib.h>
+              #include <unistd.h>
+
+              int main ()
+              {
+              #if defined(_SC_KERNEL_BITS)
+                  long bits = sysconf(_SC_KERNEL_BITS);
+              #endif
+                  long cpu  = sysconf (_SC_CPU_VERSION);
+
+                  switch (cpu)
+              	{
+              	case CPU_PA_RISC1_0: puts ("hppa1.0"); break;
+              	case CPU_PA_RISC1_1: puts ("hppa1.1"); break;
+              	case CPU_PA_RISC2_0:
+              #if defined(_SC_KERNEL_BITS)
+              	    switch (bits)
+              		{
+              		case 64: puts ("hppa2.0w"); break;
+              		case 32: puts ("hppa2.0n"); break;
+              		default: puts ("hppa2.0"); break;
+              		} break;
+              #else  /* !defined(_SC_KERNEL_BITS) */
+              	    puts ("hppa2.0"); break;
+              #endif
+              	default: puts ("hppa1.0"); break;
+              	}
+                  exit (0);
+              }
+EOF
+		    (CCOPTS= $CC_FOR_BUILD -o $dummy $dummy.c 2>/dev/null) && HP_ARCH=`$dummy`
+		    test -z "$HP_ARCH" && HP_ARCH=hppa
+		fi ;;
+	esac
+	if [ ${HP_ARCH} = "hppa2.0w" ]
+	then
+	    # avoid double evaluation of $set_cc_for_build
+	    test -n "$CC_FOR_BUILD" || eval $set_cc_for_build
+	    if echo __LP64__ | (CCOPTS= $CC_FOR_BUILD -E -) | grep __LP64__ >/dev/null
+	    then
+		HP_ARCH="hppa2.0w"
+	    else
+		HP_ARCH="hppa64"
+	    fi
+	fi
+	echo ${HP_ARCH}-hp-hpux${HPUX_REV}
+	exit 0 ;;
+    ia64:HP-UX:*:*)
+	HPUX_REV=`echo ${UNAME_RELEASE}|sed -e 's/[^.]*.[0B]*//'`
+	echo ia64-hp-hpux${HPUX_REV}
+	exit 0 ;;
+    3050*:HI-UX:*:*)
+	eval $set_cc_for_build
+	sed 's/^	//' << EOF >$dummy.c
+	#include <unistd.h>
+	int
+	main ()
+	{
+	  long cpu = sysconf (_SC_CPU_VERSION);
+	  /* The order matters, because CPU_IS_HP_MC68K erroneously returns
+	     true for CPU_PA_RISC1_0.  CPU_IS_PA_RISC returns correct
+	     results, however.  */
+	  if (CPU_IS_PA_RISC (cpu))
+	    {
+	      switch (cpu)
+		{
+		  case CPU_PA_RISC1_0: puts ("hppa1.0-hitachi-hiuxwe2"); break;
+		  case CPU_PA_RISC1_1: puts ("hppa1.1-hitachi-hiuxwe2"); break;
+		  case CPU_PA_RISC2_0: puts ("hppa2.0-hitachi-hiuxwe2"); break;
+		  default: puts ("hppa-hitachi-hiuxwe2"); break;
+		}
+	    }
+	  else if (CPU_IS_HP_MC68K (cpu))
+	    puts ("m68k-hitachi-hiuxwe2");
+	  else puts ("unknown-hitachi-hiuxwe2");
+	  exit (0);
+	}
+EOF
+	$CC_FOR_BUILD -o $dummy $dummy.c && $dummy && exit 0
+	echo unknown-hitachi-hiuxwe2
+	exit 0 ;;
+    9000/7??:4.3bsd:*:* | 9000/8?[79]:4.3bsd:*:* )
+	echo hppa1.1-hp-bsd
+	exit 0 ;;
+    9000/8??:4.3bsd:*:*)
+	echo hppa1.0-hp-bsd
+	exit 0 ;;
+    *9??*:MPE/iX:*:* | *3000*:MPE/iX:*:*)
+	echo hppa1.0-hp-mpeix
+	exit 0 ;;
+    hp7??:OSF1:*:* | hp8?[79]:OSF1:*:* )
+	echo hppa1.1-hp-osf
+	exit 0 ;;
+    hp8??:OSF1:*:*)
+	echo hppa1.0-hp-osf
+	exit 0 ;;
+    i*86:OSF1:*:*)
+	if [ -x /usr/sbin/sysversion ] ; then
+	    echo ${UNAME_MACHINE}-unknown-osf1mk
+	else
+	    echo ${UNAME_MACHINE}-unknown-osf1
+	fi
+	exit 0 ;;
+    parisc*:Lites*:*:*)
+	echo hppa1.1-hp-lites
+	exit 0 ;;
+    C1*:ConvexOS:*:* | convex:ConvexOS:C1*:*)
+	echo c1-convex-bsd
+        exit 0 ;;
+    C2*:ConvexOS:*:* | convex:ConvexOS:C2*:*)
+	if getsysinfo -f scalar_acc
+	then echo c32-convex-bsd
+	else echo c2-convex-bsd
+	fi
+        exit 0 ;;
+    C34*:ConvexOS:*:* | convex:ConvexOS:C34*:*)
+	echo c34-convex-bsd
+        exit 0 ;;
+    C38*:ConvexOS:*:* | convex:ConvexOS:C38*:*)
+	echo c38-convex-bsd
+        exit 0 ;;
+    C4*:ConvexOS:*:* | convex:ConvexOS:C4*:*)
+	echo c4-convex-bsd
+        exit 0 ;;
+    CRAY*Y-MP:*:*:*)
+	echo ymp-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/'
+	exit 0 ;;
+    CRAY*[A-Z]90:*:*:*)
+	echo ${UNAME_MACHINE}-cray-unicos${UNAME_RELEASE} \
+	| sed -e 's/CRAY.*\([A-Z]90\)/\1/' \
+	      -e y/ABCDEFGHIJKLMNOPQRSTUVWXYZ/abcdefghijklmnopqrstuvwxyz/ \
+	      -e 's/\.[^.]*$/.X/'
+	exit 0 ;;
+    CRAY*TS:*:*:*)
+	echo t90-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/'
+	exit 0 ;;
+    CRAY*T3E:*:*:*)
+	echo alphaev5-cray-unicosmk${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/'
+	exit 0 ;;
+    CRAY*SV1:*:*:*)
+	echo sv1-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/'
+	exit 0 ;;
+    *:UNICOS/mp:*:*)
+	echo nv1-cray-unicosmp${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' 
+	exit 0 ;;
+    F30[01]:UNIX_System_V:*:* | F700:UNIX_System_V:*:*)
+	FUJITSU_PROC=`uname -m | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'`
+        FUJITSU_SYS=`uname -p | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/\///'`
+        FUJITSU_REL=`echo ${UNAME_RELEASE} | sed -e 's/ /_/'`
+        echo "${FUJITSU_PROC}-fujitsu-${FUJITSU_SYS}${FUJITSU_REL}"
+        exit 0 ;;
+    i*86:BSD/386:*:* | i*86:BSD/OS:*:* | *:Ascend\ Embedded/OS:*:*)
+	echo ${UNAME_MACHINE}-pc-bsdi${UNAME_RELEASE}
+	exit 0 ;;
+    sparc*:BSD/OS:*:*)
+	echo sparc-unknown-bsdi${UNAME_RELEASE}
+	exit 0 ;;
+    *:BSD/OS:*:*)
+	echo ${UNAME_MACHINE}-unknown-bsdi${UNAME_RELEASE}
+	exit 0 ;;
+    *:FreeBSD:*:*|*:GNU/FreeBSD:*:*)
+	# Determine whether the default compiler uses glibc.
+	eval $set_cc_for_build
+	sed 's/^	//' << EOF >$dummy.c
+	#include <features.h>
+	#if __GLIBC__ >= 2
+	LIBC=gnu
+	#else
+	LIBC=
+	#endif
+EOF
+	eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^LIBC=`
+	echo ${UNAME_MACHINE}-unknown-freebsd`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'`${LIBC:+-$LIBC}
+	exit 0 ;;
+    i*:CYGWIN*:*)
+	echo ${UNAME_MACHINE}-pc-cygwin
+	exit 0 ;;
+    i*:MINGW*:*)
+	echo ${UNAME_MACHINE}-pc-mingw32
+	exit 0 ;;
+    i*:PW*:*)
+	echo ${UNAME_MACHINE}-pc-pw32
+	exit 0 ;;
+    x86:Interix*:[34]*)
+	echo i586-pc-interix${UNAME_RELEASE}|sed -e 's/\..*//'
+	exit 0 ;;
+    [345]86:Windows_95:* | [345]86:Windows_98:* | [345]86:Windows_NT:*)
+	echo i${UNAME_MACHINE}-pc-mks
+	exit 0 ;;
+    i*:Windows_NT*:* | Pentium*:Windows_NT*:*)
+	# How do we know it's Interix rather than the generic POSIX subsystem?
+	# It also conflicts with pre-2.0 versions of AT&T UWIN. Should we
+	# UNAME_MACHINE based on the output of uname instead of i386?
+	echo i586-pc-interix
+	exit 0 ;;
+    i*:UWIN*:*)
+	echo ${UNAME_MACHINE}-pc-uwin
+	exit 0 ;;
+    p*:CYGWIN*:*)
+	echo powerpcle-unknown-cygwin
+	exit 0 ;;
+    prep*:SunOS:5.*:*)
+	echo powerpcle-unknown-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'`
+	exit 0 ;;
+    *:GNU:*:*)
+	echo `echo ${UNAME_MACHINE}|sed -e 's,[-/].*$,,'`-unknown-gnu`echo ${UNAME_RELEASE}|sed -e 's,/.*$,,'`
+	exit 0 ;;
+    i*86:Minix:*:*)
+	echo ${UNAME_MACHINE}-pc-minix
+	exit 0 ;;
+    arm*:Linux:*:*)
+	echo ${UNAME_MACHINE}-unknown-linux-gnu
+	exit 0 ;;
+    cris:Linux:*:*)
+	echo cris-axis-linux-gnu
+	exit 0 ;;
+    ia64:Linux:*:*)
+	echo ${UNAME_MACHINE}-unknown-linux-gnu
+	exit 0 ;;
+    m68*:Linux:*:*)
+	echo ${UNAME_MACHINE}-unknown-linux-gnu
+	exit 0 ;;
+    mips:Linux:*:*)
+	eval $set_cc_for_build
+	sed 's/^	//' << EOF >$dummy.c
+	#undef CPU
+	#undef mips
+	#undef mipsel
+	#if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL) || defined(MIPSEL)
+	CPU=mipsel
+	#else
+	#if defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB) || defined(MIPSEB)
+	CPU=mips
+	#else
+	CPU=
+	#endif
+	#endif
+EOF
+	eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^CPU=`
+	test x"${CPU}" != x && echo "${CPU}-unknown-linux-gnu" && exit 0
+	;;
+    mips64:Linux:*:*)
+	eval $set_cc_for_build
+	sed 's/^	//' << EOF >$dummy.c
+	#undef CPU
+	#undef mips64
+	#undef mips64el
+	#if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL) || defined(MIPSEL)
+	CPU=mips64el
+	#else
+	#if defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB) || defined(MIPSEB)
+	CPU=mips64
+	#else
+	CPU=
+	#endif
+	#endif
+EOF
+	eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^CPU=`
+	test x"${CPU}" != x && echo "${CPU}-unknown-linux-gnu" && exit 0
+	;;
+    ppc:Linux:*:*)
+	echo powerpc-unknown-linux-gnu
+	exit 0 ;;
+    ppc64:Linux:*:*)
+	echo powerpc64-unknown-linux-gnu
+	exit 0 ;;
+    alpha:Linux:*:*)
+	case `sed -n '/^cpu model/s/^.*: \(.*\)/\1/p' < /proc/cpuinfo` in
+	  EV5)   UNAME_MACHINE=alphaev5 ;;
+	  EV56)  UNAME_MACHINE=alphaev56 ;;
+	  PCA56) UNAME_MACHINE=alphapca56 ;;
+	  PCA57) UNAME_MACHINE=alphapca56 ;;
+	  EV6)   UNAME_MACHINE=alphaev6 ;;
+	  EV67)  UNAME_MACHINE=alphaev67 ;;
+	  EV68*) UNAME_MACHINE=alphaev68 ;;
+        esac
+	objdump --private-headers /bin/sh | grep ld.so.1 >/dev/null
+	if test "$?" = 0 ; then LIBC="libc1" ; else LIBC="" ; fi
+	echo ${UNAME_MACHINE}-unknown-linux-gnu${LIBC}
+	exit 0 ;;
+    parisc:Linux:*:* | hppa:Linux:*:*)
+	# Look for CPU level
+	case `grep '^cpu[^a-z]*:' /proc/cpuinfo 2>/dev/null | cut -d' ' -f2` in
+	  PA7*) echo hppa1.1-unknown-linux-gnu ;;
+	  PA8*) echo hppa2.0-unknown-linux-gnu ;;
+	  *)    echo hppa-unknown-linux-gnu ;;
+	esac
+	exit 0 ;;
+    parisc64:Linux:*:* | hppa64:Linux:*:*)
+	echo hppa64-unknown-linux-gnu
+	exit 0 ;;
+    s390:Linux:*:* | s390x:Linux:*:*)
+	echo ${UNAME_MACHINE}-ibm-linux
+	exit 0 ;;
+    sh64*:Linux:*:*)
+    	echo ${UNAME_MACHINE}-unknown-linux-gnu
+	exit 0 ;;
+    sh*:Linux:*:*)
+	echo ${UNAME_MACHINE}-unknown-linux-gnu
+	exit 0 ;;
+    sparc:Linux:*:* | sparc64:Linux:*:*)
+	echo ${UNAME_MACHINE}-unknown-linux-gnu
+	exit 0 ;;
+    x86_64:Linux:*:*)
+	echo x86_64-unknown-linux-gnu
+	exit 0 ;;
+    i*86:Linux:*:*)
+	# The BFD linker knows what the default object file format is, so
+	# first see if it will tell us. cd to the root directory to prevent
+	# problems with other programs or directories called `ld' in the path.
+	# Set LC_ALL=C to ensure ld outputs messages in English.
+	ld_supported_targets=`cd /; LC_ALL=C ld --help 2>&1 \
+			 | sed -ne '/supported targets:/!d
+				    s/[ 	][ 	]*/ /g
+				    s/.*supported targets: *//
+				    s/ .*//
+				    p'`
+        case "$ld_supported_targets" in
+	  elf32-i386)
+		TENTATIVE="${UNAME_MACHINE}-pc-linux-gnu"
+		;;
+	  a.out-i386-linux)
+		echo "${UNAME_MACHINE}-pc-linux-gnuaout"
+		exit 0 ;;
+	  coff-i386)
+		echo "${UNAME_MACHINE}-pc-linux-gnucoff"
+		exit 0 ;;
+	  "")
+		# Either a pre-BFD a.out linker (linux-gnuoldld) or
+		# one that does not give us useful --help.
+		echo "${UNAME_MACHINE}-pc-linux-gnuoldld"
+		exit 0 ;;
+	esac
+	# Determine whether the default compiler is a.out or elf
+	eval $set_cc_for_build
+	sed 's/^	//' << EOF >$dummy.c
+	#include <features.h>
+	#ifdef __ELF__
+	# ifdef __GLIBC__
+	#  if __GLIBC__ >= 2
+	LIBC=gnu
+	#  else
+	LIBC=gnulibc1
+	#  endif
+	# else
+	LIBC=gnulibc1
+	# endif
+	#else
+	#ifdef __INTEL_COMPILER
+	LIBC=gnu
+	#else
+	LIBC=gnuaout
+	#endif
+	#endif
+EOF
+	eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^LIBC=`
+	test x"${LIBC}" != x && echo "${UNAME_MACHINE}-pc-linux-${LIBC}" && exit 0
+	test x"${TENTATIVE}" != x && echo "${TENTATIVE}" && exit 0
+	;;
+    i*86:DYNIX/ptx:4*:*)
+	# ptx 4.0 does uname -s correctly, with DYNIX/ptx in there.
+	# earlier versions are messed up and put the nodename in both
+	# sysname and nodename.
+	echo i386-sequent-sysv4
+	exit 0 ;;
+    i*86:UNIX_SV:4.2MP:2.*)
+        # Unixware is an offshoot of SVR4, but it has its own version
+        # number series starting with 2...
+        # I am not positive that other SVR4 systems won't match this,
+	# I just have to hope.  -- rms.
+        # Use sysv4.2uw... so that sysv4* matches it.
+	echo ${UNAME_MACHINE}-pc-sysv4.2uw${UNAME_VERSION}
+	exit 0 ;;
+    i*86:OS/2:*:*)
+	# If we were able to find `uname', then EMX Unix compatibility
+	# is probably installed.
+	echo ${UNAME_MACHINE}-pc-os2-emx
+	exit 0 ;;
+    i*86:XTS-300:*:STOP)
+	echo ${UNAME_MACHINE}-unknown-stop
+	exit 0 ;;
+    i*86:atheos:*:*)
+	echo ${UNAME_MACHINE}-unknown-atheos
+	exit 0 ;;
+    i*86:LynxOS:2.*:* | i*86:LynxOS:3.[01]*:* | i*86:LynxOS:4.0*:*)
+	echo i386-unknown-lynxos${UNAME_RELEASE}
+	exit 0 ;;
+    i*86:*DOS:*:*)
+	echo ${UNAME_MACHINE}-pc-msdosdjgpp
+	exit 0 ;;
+    i*86:*:4.*:* | i*86:SYSTEM_V:4.*:*)
+	UNAME_REL=`echo ${UNAME_RELEASE} | sed 's/\/MP$//'`
+	if grep Novell /usr/include/link.h >/dev/null 2>/dev/null; then
+		echo ${UNAME_MACHINE}-univel-sysv${UNAME_REL}
+	else
+		echo ${UNAME_MACHINE}-pc-sysv${UNAME_REL}
+	fi
+	exit 0 ;;
+    i*86:*:5:[78]*)
+	case `/bin/uname -X | grep "^Machine"` in
+	    *486*)	     UNAME_MACHINE=i486 ;;
+	    *Pentium)	     UNAME_MACHINE=i586 ;;
+	    *Pent*|*Celeron) UNAME_MACHINE=i686 ;;
+	esac
+	echo ${UNAME_MACHINE}-unknown-sysv${UNAME_RELEASE}${UNAME_SYSTEM}${UNAME_VERSION}
+	exit 0 ;;
+    i*86:*:3.2:*)
+	if test -f /usr/options/cb.name; then
+		UNAME_REL=`sed -n 's/.*Version //p' </usr/options/cb.name`
+		echo ${UNAME_MACHINE}-pc-isc$UNAME_REL
+	elif /bin/uname -X 2>/dev/null >/dev/null ; then
+		UNAME_REL=`(/bin/uname -X|grep Release|sed -e 's/.*= //')`
+		(/bin/uname -X|grep i80486 >/dev/null) && UNAME_MACHINE=i486
+		(/bin/uname -X|grep '^Machine.*Pentium' >/dev/null) \
+			&& UNAME_MACHINE=i586
+		(/bin/uname -X|grep '^Machine.*Pent *II' >/dev/null) \
+			&& UNAME_MACHINE=i686
+		(/bin/uname -X|grep '^Machine.*Pentium Pro' >/dev/null) \
+			&& UNAME_MACHINE=i686
+		echo ${UNAME_MACHINE}-pc-sco$UNAME_REL
+	else
+		echo ${UNAME_MACHINE}-pc-sysv32
+	fi
+	exit 0 ;;
+    pc:*:*:*)
+	# Left here for compatibility:
+        # uname -m prints for DJGPP always 'pc', but it prints nothing about
+        # the processor, so we play safe by assuming i386.
+	echo i386-pc-msdosdjgpp
+        exit 0 ;;
+    Intel:Mach:3*:*)
+	echo i386-pc-mach3
+	exit 0 ;;
+    paragon:*:*:*)
+	echo i860-intel-osf1
+	exit 0 ;;
+    i860:*:4.*:*) # i860-SVR4
+	if grep Stardent /usr/include/sys/uadmin.h >/dev/null 2>&1 ; then
+	  echo i860-stardent-sysv${UNAME_RELEASE} # Stardent Vistra i860-SVR4
+	else # Add other i860-SVR4 vendors below as they are discovered.
+	  echo i860-unknown-sysv${UNAME_RELEASE}  # Unknown i860-SVR4
+	fi
+	exit 0 ;;
+    mini*:CTIX:SYS*5:*)
+	# "miniframe"
+	echo m68010-convergent-sysv
+	exit 0 ;;
+    mc68k:UNIX:SYSTEM5:3.51m)
+	echo m68k-convergent-sysv
+	exit 0 ;;
+    M680?0:D-NIX:5.3:*)
+	echo m68k-diab-dnix
+	exit 0 ;;
+    M68*:*:R3V[567]*:*)
+	test -r /sysV68 && echo 'm68k-motorola-sysv' && exit 0 ;;
+    3[34]??:*:4.0:3.0 | 3[34]??A:*:4.0:3.0 | 3[34]??,*:*:4.0:3.0 | 3[34]??/*:*:4.0:3.0 | 4400:*:4.0:3.0 | 4850:*:4.0:3.0 | SKA40:*:4.0:3.0 | SDS2:*:4.0:3.0 | SHG2:*:4.0:3.0)
+	OS_REL=''
+	test -r /etc/.relid \
+	&& OS_REL=.`sed -n 's/[^ ]* [^ ]* \([0-9][0-9]\).*/\1/p' < /etc/.relid`
+	/bin/uname -p 2>/dev/null | grep 86 >/dev/null \
+	  && echo i486-ncr-sysv4.3${OS_REL} && exit 0
+	/bin/uname -p 2>/dev/null | /bin/grep entium >/dev/null \
+	  && echo i586-ncr-sysv4.3${OS_REL} && exit 0 ;;
+    3[34]??:*:4.0:* | 3[34]??,*:*:4.0:*)
+        /bin/uname -p 2>/dev/null | grep 86 >/dev/null \
+          && echo i486-ncr-sysv4 && exit 0 ;;
+    m68*:LynxOS:2.*:* | m68*:LynxOS:3.0*:*)
+	echo m68k-unknown-lynxos${UNAME_RELEASE}
+	exit 0 ;;
+    mc68030:UNIX_System_V:4.*:*)
+	echo m68k-atari-sysv4
+	exit 0 ;;
+    TSUNAMI:LynxOS:2.*:*)
+	echo sparc-unknown-lynxos${UNAME_RELEASE}
+	exit 0 ;;
+    rs6000:LynxOS:2.*:*)
+	echo rs6000-unknown-lynxos${UNAME_RELEASE}
+	exit 0 ;;
+    PowerPC:LynxOS:2.*:* | PowerPC:LynxOS:3.[01]*:* | PowerPC:LynxOS:4.0*:*)
+	echo powerpc-unknown-lynxos${UNAME_RELEASE}
+	exit 0 ;;
+    SM[BE]S:UNIX_SV:*:*)
+	echo mips-dde-sysv${UNAME_RELEASE}
+	exit 0 ;;
+    RM*:ReliantUNIX-*:*:*)
+	echo mips-sni-sysv4
+	exit 0 ;;
+    RM*:SINIX-*:*:*)
+	echo mips-sni-sysv4
+	exit 0 ;;
+    *:SINIX-*:*:*)
+	if uname -p 2>/dev/null >/dev/null ; then
+		UNAME_MACHINE=`(uname -p) 2>/dev/null`
+		echo ${UNAME_MACHINE}-sni-sysv4
+	else
+		echo ns32k-sni-sysv
+	fi
+	exit 0 ;;
+    PENTIUM:*:4.0*:*) # Unisys `ClearPath HMP IX 4000' SVR4/MP effort
+                      # says <Richard.M.Bartel at ccMail.Census.GOV>
+        echo i586-unisys-sysv4
+        exit 0 ;;
+    *:UNIX_System_V:4*:FTX*)
+	# From Gerald Hewes <hewes at openmarket.com>.
+	# How about differentiating between stratus architectures? -djm
+	echo hppa1.1-stratus-sysv4
+	exit 0 ;;
+    *:*:*:FTX*)
+	# From seanf at swdc.stratus.com.
+	echo i860-stratus-sysv4
+	exit 0 ;;
+    *:VOS:*:*)
+	# From Paul.Green at stratus.com.
+	echo hppa1.1-stratus-vos
+	exit 0 ;;
+    mc68*:A/UX:*:*)
+	echo m68k-apple-aux${UNAME_RELEASE}
+	exit 0 ;;
+    news*:NEWS-OS:6*:*)
+	echo mips-sony-newsos6
+	exit 0 ;;
+    R[34]000:*System_V*:*:* | R4000:UNIX_SYSV:*:* | R*000:UNIX_SV:*:*)
+	if [ -d /usr/nec ]; then
+	        echo mips-nec-sysv${UNAME_RELEASE}
+	else
+	        echo mips-unknown-sysv${UNAME_RELEASE}
+	fi
+        exit 0 ;;
+    BeBox:BeOS:*:*)	# BeOS running on hardware made by Be, PPC only.
+	echo powerpc-be-beos
+	exit 0 ;;
+    BeMac:BeOS:*:*)	# BeOS running on Mac or Mac clone, PPC only.
+	echo powerpc-apple-beos
+	exit 0 ;;
+    BePC:BeOS:*:*)	# BeOS running on Intel PC compatible.
+	echo i586-pc-beos
+	exit 0 ;;
+    SX-4:SUPER-UX:*:*)
+	echo sx4-nec-superux${UNAME_RELEASE}
+	exit 0 ;;
+    SX-5:SUPER-UX:*:*)
+	echo sx5-nec-superux${UNAME_RELEASE}
+	exit 0 ;;
+    SX-6:SUPER-UX:*:*)
+	echo sx6-nec-superux${UNAME_RELEASE}
+	exit 0 ;;
+    Power*:Rhapsody:*:*)
+	echo powerpc-apple-rhapsody${UNAME_RELEASE}
+	exit 0 ;;
+    *:Rhapsody:*:*)
+	echo ${UNAME_MACHINE}-apple-rhapsody${UNAME_RELEASE}
+	exit 0 ;;
+    *:Darwin:*:*)
+	case `uname -p` in
+	    *86) UNAME_PROCESSOR=i686 ;;
+	    powerpc) UNAME_PROCESSOR=powerpc ;;
+	esac
+	echo ${UNAME_PROCESSOR}-apple-darwin${UNAME_RELEASE}
+	exit 0 ;;
+    *:procnto*:*:* | *:QNX:[0123456789]*:*)
+	UNAME_PROCESSOR=`uname -p`
+	if test "$UNAME_PROCESSOR" = "x86"; then
+		UNAME_PROCESSOR=i386
+		UNAME_MACHINE=pc
+	fi
+	echo ${UNAME_PROCESSOR}-${UNAME_MACHINE}-nto-qnx${UNAME_RELEASE}
+	exit 0 ;;
+    *:QNX:*:4*)
+	echo i386-pc-qnx
+	exit 0 ;;
+    NSR-[DGKLNPTVW]:NONSTOP_KERNEL:*:*)
+	echo nsr-tandem-nsk${UNAME_RELEASE}
+	exit 0 ;;
+    *:NonStop-UX:*:*)
+	echo mips-compaq-nonstopux
+	exit 0 ;;
+    BS2000:POSIX*:*:*)
+	echo bs2000-siemens-sysv
+	exit 0 ;;
+    DS/*:UNIX_System_V:*:*)
+	echo ${UNAME_MACHINE}-${UNAME_SYSTEM}-${UNAME_RELEASE}
+	exit 0 ;;
+    *:Plan9:*:*)
+	# "uname -m" is not consistent, so use $cputype instead. 386
+	# is converted to i386 for consistency with other x86
+	# operating systems.
+	if test "$cputype" = "386"; then
+	    UNAME_MACHINE=i386
+	else
+	    UNAME_MACHINE="$cputype"
+	fi
+	echo ${UNAME_MACHINE}-unknown-plan9
+	exit 0 ;;
+    *:TOPS-10:*:*)
+	echo pdp10-unknown-tops10
+	exit 0 ;;
+    *:TENEX:*:*)
+	echo pdp10-unknown-tenex
+	exit 0 ;;
+    KS10:TOPS-20:*:* | KL10:TOPS-20:*:* | TYPE4:TOPS-20:*:*)
+	echo pdp10-dec-tops20
+	exit 0 ;;
+    XKL-1:TOPS-20:*:* | TYPE5:TOPS-20:*:*)
+	echo pdp10-xkl-tops20
+	exit 0 ;;
+    *:TOPS-20:*:*)
+	echo pdp10-unknown-tops20
+	exit 0 ;;
+    *:ITS:*:*)
+	echo pdp10-unknown-its
+	exit 0 ;;
+    SEI:*:*:SEIUX)
+        echo mips-sei-seiux${UNAME_RELEASE}
+	exit 0 ;;
+esac
+
+#echo '(No uname command or uname output not recognized.)' 1>&2
+#echo "${UNAME_MACHINE}:${UNAME_SYSTEM}:${UNAME_RELEASE}:${UNAME_VERSION}" 1>&2
+
+eval $set_cc_for_build
+cat >$dummy.c <<EOF
+#ifdef _SEQUENT_
+# include <sys/types.h>
+# include <sys/utsname.h>
+#endif
+main ()
+{
+#if defined (sony)
+#if defined (MIPSEB)
+  /* BFD wants "bsd" instead of "newsos".  Perhaps BFD should be changed,
+     I don't know....  */
+  printf ("mips-sony-bsd\n"); exit (0);
+#else
+#include <sys/param.h>
+  printf ("m68k-sony-newsos%s\n",
+#ifdef NEWSOS4
+          "4"
+#else
+	  ""
+#endif
+         ); exit (0);
+#endif
+#endif
+
+#if defined (__arm) && defined (__acorn) && defined (__unix)
+  printf ("arm-acorn-riscix"); exit (0);
+#endif
+
+#if defined (hp300) && !defined (hpux)
+  printf ("m68k-hp-bsd\n"); exit (0);
+#endif
+
+#if defined (NeXT)
+#if !defined (__ARCHITECTURE__)
+#define __ARCHITECTURE__ "m68k"
+#endif
+  int version;
+  version=`(hostinfo | sed -n 's/.*NeXT Mach \([0-9]*\).*/\1/p') 2>/dev/null`;
+  if (version < 4)
+    printf ("%s-next-nextstep%d\n", __ARCHITECTURE__, version);
+  else
+    printf ("%s-next-openstep%d\n", __ARCHITECTURE__, version);
+  exit (0);
+#endif
+
+#if defined (MULTIMAX) || defined (n16)
+#if defined (UMAXV)
+  printf ("ns32k-encore-sysv\n"); exit (0);
+#else
+#if defined (CMU)
+  printf ("ns32k-encore-mach\n"); exit (0);
+#else
+  printf ("ns32k-encore-bsd\n"); exit (0);
+#endif
+#endif
+#endif
+
+#if defined (__386BSD__)
+  printf ("i386-pc-bsd\n"); exit (0);
+#endif
+
+#if defined (sequent)
+#if defined (i386)
+  printf ("i386-sequent-dynix\n"); exit (0);
+#endif
+#if defined (ns32000)
+  printf ("ns32k-sequent-dynix\n"); exit (0);
+#endif
+#endif
+
+#if defined (_SEQUENT_)
+    struct utsname un;
+
+    uname(&un);
+
+    if (strncmp(un.version, "V2", 2) == 0) {
+	printf ("i386-sequent-ptx2\n"); exit (0);
+    }
+    if (strncmp(un.version, "V1", 2) == 0) { /* XXX is V1 correct? */
+	printf ("i386-sequent-ptx1\n"); exit (0);
+    }
+    printf ("i386-sequent-ptx\n"); exit (0);
+
+#endif
+
+#if defined (vax)
+# if !defined (ultrix)
+#  include <sys/param.h>
+#  if defined (BSD)
+#   if BSD == 43
+      printf ("vax-dec-bsd4.3\n"); exit (0);
+#   else
+#    if BSD == 199006
+      printf ("vax-dec-bsd4.3reno\n"); exit (0);
+#    else
+      printf ("vax-dec-bsd\n"); exit (0);
+#    endif
+#   endif
+#  else
+    printf ("vax-dec-bsd\n"); exit (0);
+#  endif
+# else
+    printf ("vax-dec-ultrix\n"); exit (0);
+# endif
+#endif
+
+#if defined (alliant) && defined (i860)
+  printf ("i860-alliant-bsd\n"); exit (0);
+#endif
+
+  exit (1);
+}
+EOF
+
+$CC_FOR_BUILD -o $dummy $dummy.c 2>/dev/null && $dummy && exit 0
+
+# Apollos put the system type in the environment.
+
+test -d /usr/apollo && { echo ${ISP}-apollo-${SYSTYPE}; exit 0; }
+
+# Convex versions that predate uname can use getsysinfo(1)
+
+if [ -x /usr/convex/getsysinfo ]
+then
+    case `getsysinfo -f cpu_type` in
+    c1*)
+	echo c1-convex-bsd
+	exit 0 ;;
+    c2*)
+	if getsysinfo -f scalar_acc
+	then echo c32-convex-bsd
+	else echo c2-convex-bsd
+	fi
+	exit 0 ;;
+    c34*)
+	echo c34-convex-bsd
+	exit 0 ;;
+    c38*)
+	echo c38-convex-bsd
+	exit 0 ;;
+    c4*)
+	echo c4-convex-bsd
+	exit 0 ;;
+    esac
+fi
+
+cat >&2 <<EOF
+$0: unable to guess system type
+
+This script, last modified $timestamp, has failed to recognize
+the operating system you are using. It is advised that you
+download the most up to date version of the config scripts from
+
+    ftp://ftp.gnu.org/pub/gnu/config/
+
+If the version you run ($0) is already up to date, please
+send the following data and any information you think might be
+pertinent to <config-patches at gnu.org> in order to provide the needed
+information to handle your system.
+
+config.guess timestamp = $timestamp
+
+uname -m = `(uname -m) 2>/dev/null || echo unknown`
+uname -r = `(uname -r) 2>/dev/null || echo unknown`
+uname -s = `(uname -s) 2>/dev/null || echo unknown`
+uname -v = `(uname -v) 2>/dev/null || echo unknown`
+
+/usr/bin/uname -p = `(/usr/bin/uname -p) 2>/dev/null`
+/bin/uname -X     = `(/bin/uname -X) 2>/dev/null`
+
+hostinfo               = `(hostinfo) 2>/dev/null`
+/bin/universe          = `(/bin/universe) 2>/dev/null`
+/usr/bin/arch -k       = `(/usr/bin/arch -k) 2>/dev/null`
+/bin/arch              = `(/bin/arch) 2>/dev/null`
+/usr/bin/oslevel       = `(/usr/bin/oslevel) 2>/dev/null`
+/usr/convex/getsysinfo = `(/usr/convex/getsysinfo) 2>/dev/null`
+
+UNAME_MACHINE = ${UNAME_MACHINE}
+UNAME_RELEASE = ${UNAME_RELEASE}
+UNAME_SYSTEM  = ${UNAME_SYSTEM}
+UNAME_VERSION = ${UNAME_VERSION}
+EOF
+
+exit 1
+
+# Local variables:
+# eval: (add-hook 'write-file-hooks 'time-stamp)
+# time-stamp-start: "timestamp='"
+# time-stamp-format: "%:y-%02m-%02d"
+# time-stamp-end: "'"
+# End:
/cvs/dm/dmraid/autoconf/config.sub,v  -->  standard output
revision 1.1
--- dmraid/autoconf/config.sub
+++ -	2008-02-22 16:57:37.920041000 +0000
@@ -0,0 +1,1504 @@
+#! /bin/sh
+# Configuration validation subroutine script.
+#   Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999,
+#   2000, 2001, 2002, 2003 Free Software Foundation, Inc.
+
+timestamp='2003-06-17'
+
+# This file is (in principle) common to ALL GNU software.
+# The presence of a machine in this file suggests that SOME GNU software
+# can handle that machine.  It does not imply ALL GNU software can.
+#
+# This file is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330,
+# Boston, MA 02111-1307, USA.
+
+# As a special exception to the GNU General Public License, if you
+# distribute this file as part of a program that contains a
+# configuration script generated by Autoconf, you may include it under
+# the same distribution terms that you use for the rest of that program.
+
+# Please send patches to <config-patches at gnu.org>.  Submit a context
+# diff and a properly formatted ChangeLog entry.
+#
+# Configuration subroutine to validate and canonicalize a configuration type.
+# Supply the specified configuration type as an argument.
+# If it is invalid, we print an error message on stderr and exit with code 1.
+# Otherwise, we print the canonical config type on stdout and succeed.
+
+# This file is supposed to be the same for all GNU packages
+# and recognize all the CPU types, system types and aliases
+# that are meaningful with *any* GNU software.
+# Each package is responsible for reporting which valid configurations
+# it does not support.  The user should be able to distinguish
+# a failure to support a valid configuration from a meaningless
+# configuration.
+
+# The goal of this file is to map all the various variations of a given
+# machine specification into a single specification in the form:
+#	CPU_TYPE-MANUFACTURER-OPERATING_SYSTEM
+# or in some cases, the newer four-part form:
+#	CPU_TYPE-MANUFACTURER-KERNEL-OPERATING_SYSTEM
+# It is wrong to echo any other type of specification.
+
+me=`echo "$0" | sed -e 's,.*/,,'`
+
+usage="\
+Usage: $0 [OPTION] CPU-MFR-OPSYS
+       $0 [OPTION] ALIAS
+
+Canonicalize a configuration name.
+
+Operation modes:
+  -h, --help         print this help, then exit
+  -t, --time-stamp   print date of last modification, then exit
+  -v, --version      print version number, then exit
+
+Report bugs and patches to <config-patches at gnu.org>."
+
+version="\
+GNU config.sub ($timestamp)
+
+Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001
+Free Software Foundation, Inc.
+
+This is free software; see the source for copying conditions.  There is NO
+warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE."
+
+help="
+Try \`$me --help' for more information."
+
+# Parse command line
+while test $# -gt 0 ; do
+  case $1 in
+    --time-stamp | --time* | -t )
+       echo "$timestamp" ; exit 0 ;;
+    --version | -v )
+       echo "$version" ; exit 0 ;;
+    --help | --h* | -h )
+       echo "$usage"; exit 0 ;;
+    -- )     # Stop option processing
+       shift; break ;;
+    - )	# Use stdin as input.
+       break ;;
+    -* )
+       echo "$me: invalid option $1$help"
+       exit 1 ;;
+
+    *local*)
+       # First pass through any local machine types.
+       echo $1
+       exit 0;;
+
+    * )
+       break ;;
+  esac
+done
+
+case $# in
+ 0) echo "$me: missing argument$help" >&2
+    exit 1;;
+ 1) ;;
+ *) echo "$me: too many arguments$help" >&2
+    exit 1;;
+esac
+
+# Separate what the user gave into CPU-COMPANY and OS or KERNEL-OS (if any).
+# Here we must recognize all the valid KERNEL-OS combinations.
+maybe_os=`echo $1 | sed 's/^\(.*\)-\([^-]*-[^-]*\)$/\2/'`
+case $maybe_os in
+  nto-qnx* | linux-gnu* | freebsd*-gnu* | netbsd*-gnu* | storm-chaos* | os2-emx* | rtmk-nova*)
+    os=-$maybe_os
+    basic_machine=`echo $1 | sed 's/^\(.*\)-\([^-]*-[^-]*\)$/\1/'`
+    ;;
+  *)
+    basic_machine=`echo $1 | sed 's/-[^-]*$//'`
+    if [ $basic_machine != $1 ]
+    then os=`echo $1 | sed 's/.*-/-/'`
+    else os=; fi
+    ;;
+esac
+
+### Let's recognize common machines as not being operating systems so
+### that things like config.sub decstation-3100 work.  We also
+### recognize some manufacturers as not being operating systems, so we
+### can provide default operating systems below.
+case $os in
+	-sun*os*)
+		# Prevent following clause from handling this invalid input.
+		;;
+	-dec* | -mips* | -sequent* | -encore* | -pc532* | -sgi* | -sony* | \
+	-att* | -7300* | -3300* | -delta* | -motorola* | -sun[234]* | \
+	-unicom* | -ibm* | -next | -hp | -isi* | -apollo | -altos* | \
+	-convergent* | -ncr* | -news | -32* | -3600* | -3100* | -hitachi* |\
+	-c[123]* | -convex* | -sun | -crds | -omron* | -dg | -ultra | -tti* | \
+	-harris | -dolphin | -highlevel | -gould | -cbm | -ns | -masscomp | \
+	-apple | -axis)
+		os=
+		basic_machine=$1
+		;;
+	-sim | -cisco | -oki | -wec | -winbond)
+		os=
+		basic_machine=$1
+		;;
+	-scout)
+		;;
+	-wrs)
+		os=-vxworks
+		basic_machine=$1
+		;;
+	-chorusos*)
+		os=-chorusos
+		basic_machine=$1
+		;;
+ 	-chorusrdb)
+ 		os=-chorusrdb
+		basic_machine=$1
+ 		;;
+	-hiux*)
+		os=-hiuxwe2
+		;;
+	-sco5)
+		os=-sco3.2v5
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-sco4)
+		os=-sco3.2v4
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-sco3.2.[4-9]*)
+		os=`echo $os | sed -e 's/sco3.2./sco3.2v/'`
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-sco3.2v[4-9]*)
+		# Don't forget version if it is 3.2v4 or newer.
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-sco*)
+		os=-sco3.2v2
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-udk*)
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-isc)
+		os=-isc2.2
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-clix*)
+		basic_machine=clipper-intergraph
+		;;
+	-isc*)
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'`
+		;;
+	-lynx*)
+		os=-lynxos
+		;;
+	-ptx*)
+		basic_machine=`echo $1 | sed -e 's/86-.*/86-sequent/'`
+		;;
+	-windowsnt*)
+		os=`echo $os | sed -e 's/windowsnt/winnt/'`
+		;;
+	-psos*)
+		os=-psos
+		;;
+	-mint | -mint[0-9]*)
+		basic_machine=m68k-atari
+		os=-mint
+		;;
+esac
+
+# Decode aliases for certain CPU-COMPANY combinations.
+case $basic_machine in
+	# Recognize the basic CPU types without company name.
+	# Some are omitted here because they have special meanings below.
+	1750a | 580 \
+	| a29k \
+	| alpha | alphaev[4-8] | alphaev56 | alphaev6[78] | alphapca5[67] \
+	| alpha64 | alpha64ev[4-8] | alpha64ev56 | alpha64ev6[78] | alpha64pca5[67] \
+	| arc | arm | arm[bl]e | arme[lb] | armv[2345] | armv[345][lb] | avr \
+	| c4x | clipper \
+	| d10v | d30v | dlx | dsp16xx \
+	| fr30 | frv \
+	| h8300 | h8500 | hppa | hppa1.[01] | hppa2.0 | hppa2.0[nw] | hppa64 \
+	| i370 | i860 | i960 | ia64 \
+	| ip2k \
+	| m32r | m68000 | m68k | m88k | mcore \
+	| mips | mipsbe | mipseb | mipsel | mipsle \
+	| mips16 \
+	| mips64 | mips64el \
+	| mips64vr | mips64vrel \
+	| mips64orion | mips64orionel \
+	| mips64vr4100 | mips64vr4100el \
+	| mips64vr4300 | mips64vr4300el \
+	| mips64vr5000 | mips64vr5000el \
+	| mipsisa32 | mipsisa32el \
+	| mipsisa32r2 | mipsisa32r2el \
+	| mipsisa64 | mipsisa64el \
+	| mipsisa64sb1 | mipsisa64sb1el \
+	| mipsisa64sr71k | mipsisa64sr71kel \
+	| mipstx39 | mipstx39el \
+	| mn10200 | mn10300 \
+	| msp430 \
+	| ns16k | ns32k \
+	| openrisc | or32 \
+	| pdp10 | pdp11 | pj | pjl \
+	| powerpc | powerpc64 | powerpc64le | powerpcle | ppcbe \
+	| pyramid \
+	| sh | sh[1234] | sh[23]e | sh[34]eb | shbe | shle | sh[1234]le | sh3ele \
+	| sh64 | sh64le \
+	| sparc | sparc64 | sparc86x | sparclet | sparclite | sparcv9 | sparcv9b \
+	| strongarm \
+	| tahoe | thumb | tic4x | tic80 | tron \
+	| v850 | v850e \
+	| we32k \
+	| x86 | xscale | xstormy16 | xtensa \
+	| z8k)
+		basic_machine=$basic_machine-unknown
+		;;
+	m6811 | m68hc11 | m6812 | m68hc12)
+		# Motorola 68HC11/12.
+		basic_machine=$basic_machine-unknown
+		os=-none
+		;;
+	m88110 | m680[12346]0 | m683?2 | m68360 | m5200 | v70 | w65 | z8k)
+		;;
+
+	# We use `pc' rather than `unknown'
+	# because (1) that's what they normally are, and
+	# (2) the word "unknown" tends to confuse beginning users.
+	i*86 | x86_64)
+	  basic_machine=$basic_machine-pc
+	  ;;
+	# Object if more than one company name word.
+	*-*-*)
+		echo Invalid configuration \`$1\': machine \`$basic_machine\' not recognized 1>&2
+		exit 1
+		;;
+	# Recognize the basic CPU types with company name.
+	580-* \
+	| a29k-* \
+	| alpha-* | alphaev[4-8]-* | alphaev56-* | alphaev6[78]-* \
+	| alpha64-* | alpha64ev[4-8]-* | alpha64ev56-* | alpha64ev6[78]-* \
+	| alphapca5[67]-* | alpha64pca5[67]-* | arc-* \
+	| arm-*  | armbe-* | armle-* | armeb-* | armv*-* \
+	| avr-* \
+	| bs2000-* \
+	| c[123]* | c30-* | [cjt]90-* | c4x-* | c54x-* | c55x-* | c6x-* \
+	| clipper-* | cydra-* \
+	| d10v-* | d30v-* | dlx-* \
+	| elxsi-* \
+	| f30[01]-* | f700-* | fr30-* | frv-* | fx80-* \
+	| h8300-* | h8500-* \
+	| hppa-* | hppa1.[01]-* | hppa2.0-* | hppa2.0[nw]-* | hppa64-* \
+	| i*86-* | i860-* | i960-* | ia64-* \
+	| ip2k-* \
+	| m32r-* \
+	| m68000-* | m680[012346]0-* | m68360-* | m683?2-* | m68k-* \
+	| m88110-* | m88k-* | mcore-* \
+	| mips-* | mipsbe-* | mipseb-* | mipsel-* | mipsle-* \
+	| mips16-* \
+	| mips64-* | mips64el-* \
+	| mips64vr-* | mips64vrel-* \
+	| mips64orion-* | mips64orionel-* \
+	| mips64vr4100-* | mips64vr4100el-* \
+	| mips64vr4300-* | mips64vr4300el-* \
+	| mips64vr5000-* | mips64vr5000el-* \
+	| mipsisa32-* | mipsisa32el-* \
+	| mipsisa32r2-* | mipsisa32r2el-* \
+	| mipsisa64-* | mipsisa64el-* \
+	| mipsisa64sb1-* | mipsisa64sb1el-* \
+	| mipsisa64sr71k-* | mipsisa64sr71kel-* \
+	| mipstx39-* | mipstx39el-* \
+	| msp430-* \
+	| none-* | np1-* | nv1-* | ns16k-* | ns32k-* \
+	| orion-* \
+	| pdp10-* | pdp11-* | pj-* | pjl-* | pn-* | power-* \
+	| powerpc-* | powerpc64-* | powerpc64le-* | powerpcle-* | ppcbe-* \
+	| pyramid-* \
+	| romp-* | rs6000-* \
+	| sh-* | sh[1234]-* | sh[23]e-* | sh[34]eb-* | shbe-* \
+	| shle-* | sh[1234]le-* | sh3ele-* | sh64-* | sh64le-* \
+	| sparc-* | sparc64-* | sparc86x-* | sparclet-* | sparclite-* \
+	| sparcv9-* | sparcv9b-* | strongarm-* | sv1-* | sx?-* \
+	| tahoe-* | thumb-* \
+	| tic30-* | tic4x-* | tic54x-* | tic55x-* | tic6x-* | tic80-* \
+	| tron-* \
+	| v850-* | v850e-* | vax-* \
+	| we32k-* \
+	| x86-* | x86_64-* | xps100-* | xscale-* | xstormy16-* \
+	| xtensa-* \
+	| ymp-* \
+	| z8k-*)
+		;;
+	# Recognize the various machine names and aliases which stand
+	# for a CPU type and a company and sometimes even an OS.
+	386bsd)
+		basic_machine=i386-unknown
+		os=-bsd
+		;;
+	3b1 | 7300 | 7300-att | att-7300 | pc7300 | safari | unixpc)
+		basic_machine=m68000-att
+		;;
+	3b*)
+		basic_machine=we32k-att
+		;;
+	a29khif)
+		basic_machine=a29k-amd
+		os=-udi
+		;;
+	adobe68k)
+		basic_machine=m68010-adobe
+		os=-scout
+		;;
+	alliant | fx80)
+		basic_machine=fx80-alliant
+		;;
+	altos | altos3068)
+		basic_machine=m68k-altos
+		;;
+	am29k)
+		basic_machine=a29k-none
+		os=-bsd
+		;;
+	amd64)
+		basic_machine=x86_64-pc
+		;;
+	amdahl)
+		basic_machine=580-amdahl
+		os=-sysv
+		;;
+	amiga | amiga-*)
+		basic_machine=m68k-unknown
+		;;
+	amigaos | amigados)
+		basic_machine=m68k-unknown
+		os=-amigaos
+		;;
+	amigaunix | amix)
+		basic_machine=m68k-unknown
+		os=-sysv4
+		;;
+	apollo68)
+		basic_machine=m68k-apollo
+		os=-sysv
+		;;
+	apollo68bsd)
+		basic_machine=m68k-apollo
+		os=-bsd
+		;;
+	aux)
+		basic_machine=m68k-apple
+		os=-aux
+		;;
+	balance)
+		basic_machine=ns32k-sequent
+		os=-dynix
+		;;
+	c90)
+		basic_machine=c90-cray
+		os=-unicos
+		;;
+	convex-c1)
+		basic_machine=c1-convex
+		os=-bsd
+		;;
+	convex-c2)
+		basic_machine=c2-convex
+		os=-bsd
+		;;
+	convex-c32)
+		basic_machine=c32-convex
+		os=-bsd
+		;;
+	convex-c34)
+		basic_machine=c34-convex
+		os=-bsd
+		;;
+	convex-c38)
+		basic_machine=c38-convex
+		os=-bsd
+		;;
+	cray | j90)
+		basic_machine=j90-cray
+		os=-unicos
+		;;
+	crds | unos)
+		basic_machine=m68k-crds
+		;;
+	cris | cris-* | etrax*)
+		basic_machine=cris-axis
+		;;
+	da30 | da30-*)
+		basic_machine=m68k-da30
+		;;
+	decstation | decstation-3100 | pmax | pmax-* | pmin | dec3100 | decstatn)
+		basic_machine=mips-dec
+		;;
+	decsystem10* | dec10*)
+		basic_machine=pdp10-dec
+		os=-tops10
+		;;
+	decsystem20* | dec20*)
+		basic_machine=pdp10-dec
+		os=-tops20
+		;;
+	delta | 3300 | motorola-3300 | motorola-delta \
+	      | 3300-motorola | delta-motorola)
+		basic_machine=m68k-motorola
+		;;
+	delta88)
+		basic_machine=m88k-motorola
+		os=-sysv3
+		;;
+	dpx20 | dpx20-*)
+		basic_machine=rs6000-bull
+		os=-bosx
+		;;
+	dpx2* | dpx2*-bull)
+		basic_machine=m68k-bull
+		os=-sysv3
+		;;
+	ebmon29k)
+		basic_machine=a29k-amd
+		os=-ebmon
+		;;
+	elxsi)
+		basic_machine=elxsi-elxsi
+		os=-bsd
+		;;
+	encore | umax | mmax)
+		basic_machine=ns32k-encore
+		;;
+	es1800 | OSE68k | ose68k | ose | OSE)
+		basic_machine=m68k-ericsson
+		os=-ose
+		;;
+	fx2800)
+		basic_machine=i860-alliant
+		;;
+	genix)
+		basic_machine=ns32k-ns
+		;;
+	gmicro)
+		basic_machine=tron-gmicro
+		os=-sysv
+		;;
+	go32)
+		basic_machine=i386-pc
+		os=-go32
+		;;
+	h3050r* | hiux*)
+		basic_machine=hppa1.1-hitachi
+		os=-hiuxwe2
+		;;
+	h8300hms)
+		basic_machine=h8300-hitachi
+		os=-hms
+		;;
+	h8300xray)
+		basic_machine=h8300-hitachi
+		os=-xray
+		;;
+	h8500hms)
+		basic_machine=h8500-hitachi
+		os=-hms
+		;;
+	harris)
+		basic_machine=m88k-harris
+		os=-sysv3
+		;;
+	hp300-*)
+		basic_machine=m68k-hp
+		;;
+	hp300bsd)
+		basic_machine=m68k-hp
+		os=-bsd
+		;;
+	hp300hpux)
+		basic_machine=m68k-hp
+		os=-hpux
+		;;
+	hp3k9[0-9][0-9] | hp9[0-9][0-9])
+		basic_machine=hppa1.0-hp
+		;;
+	hp9k2[0-9][0-9] | hp9k31[0-9])
+		basic_machine=m68000-hp
+		;;
+	hp9k3[2-9][0-9])
+		basic_machine=m68k-hp
+		;;
+	hp9k6[0-9][0-9] | hp6[0-9][0-9])
+		basic_machine=hppa1.0-hp
+		;;
+	hp9k7[0-79][0-9] | hp7[0-79][0-9])
+		basic_machine=hppa1.1-hp
+		;;
+	hp9k78[0-9] | hp78[0-9])
+		# FIXME: really hppa2.0-hp
+		basic_machine=hppa1.1-hp
+		;;
+	hp9k8[67]1 | hp8[67]1 | hp9k80[24] | hp80[24] | hp9k8[78]9 | hp8[78]9 | hp9k893 | hp893)
+		# FIXME: really hppa2.0-hp
+		basic_machine=hppa1.1-hp
+		;;
+	hp9k8[0-9][13679] | hp8[0-9][13679])
+		basic_machine=hppa1.1-hp
+		;;
+	hp9k8[0-9][0-9] | hp8[0-9][0-9])
+		basic_machine=hppa1.0-hp
+		;;
+	hppa-next)
+		os=-nextstep3
+		;;
+	hppaosf)
+		basic_machine=hppa1.1-hp
+		os=-osf
+		;;
+	hppro)
+		basic_machine=hppa1.1-hp
+		os=-proelf
+		;;
+	i370-ibm* | ibm*)
+		basic_machine=i370-ibm
+		;;
+# I'm not sure what "Sysv32" means.  Should this be sysv3.2?
+	i*86v32)
+		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'`
+		os=-sysv32
+		;;
+	i*86v4*)
+		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'`
+		os=-sysv4
+		;;
+	i*86v)
+		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'`
+		os=-sysv
+		;;
+	i*86sol2)
+		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'`
+		os=-solaris2
+		;;
+	i386mach)
+		basic_machine=i386-mach
+		os=-mach
+		;;
+	i386-vsta | vsta)
+		basic_machine=i386-unknown
+		os=-vsta
+		;;
+	iris | iris4d)
+		basic_machine=mips-sgi
+		case $os in
+		    -irix*)
+			;;
+		    *)
+			os=-irix4
+			;;
+		esac
+		;;
+	isi68 | isi)
+		basic_machine=m68k-isi
+		os=-sysv
+		;;
+	m88k-omron*)
+		basic_machine=m88k-omron
+		;;
+	magnum | m3230)
+		basic_machine=mips-mips
+		os=-sysv
+		;;
+	merlin)
+		basic_machine=ns32k-utek
+		os=-sysv
+		;;
+	mingw32)
+		basic_machine=i386-pc
+		os=-mingw32
+		;;
+	miniframe)
+		basic_machine=m68000-convergent
+		;;
+	*mint | -mint[0-9]* | *MiNT | *MiNT[0-9]*)
+		basic_machine=m68k-atari
+		os=-mint
+		;;
+	mips3*-*)
+		basic_machine=`echo $basic_machine | sed -e 's/mips3/mips64/'`
+		;;
+	mips3*)
+		basic_machine=`echo $basic_machine | sed -e 's/mips3/mips64/'`-unknown
+		;;
+	mmix*)
+		basic_machine=mmix-knuth
+		os=-mmixware
+		;;
+	monitor)
+		basic_machine=m68k-rom68k
+		os=-coff
+		;;
+	morphos)
+		basic_machine=powerpc-unknown
+		os=-morphos
+		;;
+	msdos)
+		basic_machine=i386-pc
+		os=-msdos
+		;;
+	mvs)
+		basic_machine=i370-ibm
+		os=-mvs
+		;;
+	ncr3000)
+		basic_machine=i486-ncr
+		os=-sysv4
+		;;
+	netbsd386)
+		basic_machine=i386-unknown
+		os=-netbsd
+		;;
+	netwinder)
+		basic_machine=armv4l-rebel
+		os=-linux
+		;;
+	news | news700 | news800 | news900)
+		basic_machine=m68k-sony
+		os=-newsos
+		;;
+	news1000)
+		basic_machine=m68030-sony
+		os=-newsos
+		;;
+	news-3600 | risc-news)
+		basic_machine=mips-sony
+		os=-newsos
+		;;
+	necv70)
+		basic_machine=v70-nec
+		os=-sysv
+		;;
+	next | m*-next )
+		basic_machine=m68k-next
+		case $os in
+		    -nextstep* )
+			;;
+		    -ns2*)
+		      os=-nextstep2
+			;;
+		    *)
+		      os=-nextstep3
+			;;
+		esac
+		;;
+	nh3000)
+		basic_machine=m68k-harris
+		os=-cxux
+		;;
+	nh[45]000)
+		basic_machine=m88k-harris
+		os=-cxux
+		;;
+	nindy960)
+		basic_machine=i960-intel
+		os=-nindy
+		;;
+	mon960)
+		basic_machine=i960-intel
+		os=-mon960
+		;;
+	nonstopux)
+		basic_machine=mips-compaq
+		os=-nonstopux
+		;;
+	np1)
+		basic_machine=np1-gould
+		;;
+	nv1)
+		basic_machine=nv1-cray
+		os=-unicosmp
+		;;
+	nsr-tandem)
+		basic_machine=nsr-tandem
+		;;
+	op50n-* | op60c-*)
+		basic_machine=hppa1.1-oki
+		os=-proelf
+		;;
+	or32 | or32-*)
+		basic_machine=or32-unknown
+		os=-coff
+		;;
+	OSE68000 | ose68000)
+		basic_machine=m68000-ericsson
+		os=-ose
+		;;
+	os68k)
+		basic_machine=m68k-none
+		os=-os68k
+		;;
+	pa-hitachi)
+		basic_machine=hppa1.1-hitachi
+		os=-hiuxwe2
+		;;
+	paragon)
+		basic_machine=i860-intel
+		os=-osf
+		;;
+	pbd)
+		basic_machine=sparc-tti
+		;;
+	pbb)
+		basic_machine=m68k-tti
+		;;
+	pc532 | pc532-*)
+		basic_machine=ns32k-pc532
+		;;
+	pentium | p5 | k5 | k6 | nexgen | viac3)
+		basic_machine=i586-pc
+		;;
+	pentiumpro | p6 | 6x86 | athlon | athlon_*)
+		basic_machine=i686-pc
+		;;
+	pentiumii | pentium2 | pentiumiii | pentium3)
+		basic_machine=i686-pc
+		;;
+	pentium4)
+		basic_machine=i786-pc
+		;;
+	pentium-* | p5-* | k5-* | k6-* | nexgen-* | viac3-*)
+		basic_machine=i586-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	pentiumpro-* | p6-* | 6x86-* | athlon-*)
+		basic_machine=i686-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	pentiumii-* | pentium2-* | pentiumiii-* | pentium3-*)
+		basic_machine=i686-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	pentium4-*)
+		basic_machine=i786-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	pn)
+		basic_machine=pn-gould
+		;;
+	power)	basic_machine=power-ibm
+		;;
+	ppc)	basic_machine=powerpc-unknown
+		;;
+	ppc-*)	basic_machine=powerpc-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	ppcle | powerpclittle | ppc-le | powerpc-little)
+		basic_machine=powerpcle-unknown
+		;;
+	ppcle-* | powerpclittle-*)
+		basic_machine=powerpcle-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	ppc64)	basic_machine=powerpc64-unknown
+		;;
+	ppc64-*) basic_machine=powerpc64-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	ppc64le | powerpc64little | ppc64-le | powerpc64-little)
+		basic_machine=powerpc64le-unknown
+		;;
+	ppc64le-* | powerpc64little-*)
+		basic_machine=powerpc64le-`echo $basic_machine | sed 's/^[^-]*-//'`
+		;;
+	ps2)
+		basic_machine=i386-ibm
+		;;
+	pw32)
+		basic_machine=i586-unknown
+		os=-pw32
+		;;
+	rom68k)
+		basic_machine=m68k-rom68k
+		os=-coff
+		;;
+	rm[46]00)
+		basic_machine=mips-siemens
+		;;
+	rtpc | rtpc-*)
+		basic_machine=romp-ibm
+		;;
+	s390 | s390-*)
+		basic_machine=s390-ibm
+		;;
+	s390x | s390x-*)
+		basic_machine=s390x-ibm
+		;;
+	sa29200)
+		basic_machine=a29k-amd
+		os=-udi
+		;;
+	sb1)
+		basic_machine=mipsisa64sb1-unknown
+		;;
+	sb1el)
+		basic_machine=mipsisa64sb1el-unknown
+		;;
+	sei)
+		basic_machine=mips-sei
+		os=-seiux
+		;;
+	sequent)
+		basic_machine=i386-sequent
+		;;
+	sh)
+		basic_machine=sh-hitachi
+		os=-hms
+		;;
+	sh64)
+		basic_machine=sh64-unknown
+		;;
+	sparclite-wrs | simso-wrs)
+		basic_machine=sparclite-wrs
+		os=-vxworks
+		;;
+	sps7)
+		basic_machine=m68k-bull
+		os=-sysv2
+		;;
+	spur)
+		basic_machine=spur-unknown
+		;;
+	st2000)
+		basic_machine=m68k-tandem
+		;;
+	stratus)
+		basic_machine=i860-stratus
+		os=-sysv4
+		;;
+	sun2)
+		basic_machine=m68000-sun
+		;;
+	sun2os3)
+		basic_machine=m68000-sun
+		os=-sunos3
+		;;
+	sun2os4)
+		basic_machine=m68000-sun
+		os=-sunos4
+		;;
+	sun3os3)
+		basic_machine=m68k-sun
+		os=-sunos3
+		;;
+	sun3os4)
+		basic_machine=m68k-sun
+		os=-sunos4
+		;;
+	sun4os3)
+		basic_machine=sparc-sun
+		os=-sunos3
+		;;
+	sun4os4)
+		basic_machine=sparc-sun
+		os=-sunos4
+		;;
+	sun4sol2)
+		basic_machine=sparc-sun
+		os=-solaris2
+		;;
+	sun3 | sun3-*)
+		basic_machine=m68k-sun
+		;;
+	sun4)
+		basic_machine=sparc-sun
+		;;
+	sun386 | sun386i | roadrunner)
+		basic_machine=i386-sun
+		;;
+	sv1)
+		basic_machine=sv1-cray
+		os=-unicos
+		;;
+	symmetry)
+		basic_machine=i386-sequent
+		os=-dynix
+		;;
+	t3e)
+		basic_machine=alphaev5-cray
+		os=-unicos
+		;;
+	t90)
+		basic_machine=t90-cray
+		os=-unicos
+		;;
+	tic54x | c54x*)
+		basic_machine=tic54x-unknown
+		os=-coff
+		;;
+	tic55x | c55x*)
+		basic_machine=tic55x-unknown
+		os=-coff
+		;;
+	tic6x | c6x*)
+		basic_machine=tic6x-unknown
+		os=-coff
+		;;
+	tx39)
+		basic_machine=mipstx39-unknown
+		;;
+	tx39el)
+		basic_machine=mipstx39el-unknown
+		;;
+	toad1)
+		basic_machine=pdp10-xkl
+		os=-tops20
+		;;
+	tower | tower-32)
+		basic_machine=m68k-ncr
+		;;
+	udi29k)
+		basic_machine=a29k-amd
+		os=-udi
+		;;
+	ultra3)
+		basic_machine=a29k-nyu
+		os=-sym1
+		;;
+	v810 | necv810)
+		basic_machine=v810-nec
+		os=-none
+		;;
+	vaxv)
+		basic_machine=vax-dec
+		os=-sysv
+		;;
+	vms)
+		basic_machine=vax-dec
+		os=-vms
+		;;
+	vpp*|vx|vx-*)
+		basic_machine=f301-fujitsu
+		;;
+	vxworks960)
+		basic_machine=i960-wrs
+		os=-vxworks
+		;;
+	vxworks68)
+		basic_machine=m68k-wrs
+		os=-vxworks
+		;;
+	vxworks29k)
+		basic_machine=a29k-wrs
+		os=-vxworks
+		;;
+	w65*)
+		basic_machine=w65-wdc
+		os=-none
+		;;
+	w89k-*)
+		basic_machine=hppa1.1-winbond
+		os=-proelf
+		;;
+	xps | xps100)
+		basic_machine=xps100-honeywell
+		;;
+	ymp)
+		basic_machine=ymp-cray
+		os=-unicos
+		;;
+	z8k-*-coff)
+		basic_machine=z8k-unknown
+		os=-sim
+		;;
+	none)
+		basic_machine=none-none
+		os=-none
+		;;
+
+# Here we handle the default manufacturer of certain CPU types.  It is in
+# some cases the only manufacturer, in others, it is the most popular.
+	w89k)
+		basic_machine=hppa1.1-winbond
+		;;
+	op50n)
+		basic_machine=hppa1.1-oki
+		;;
+	op60c)
+		basic_machine=hppa1.1-oki
+		;;
+	romp)
+		basic_machine=romp-ibm
+		;;
+	rs6000)
+		basic_machine=rs6000-ibm
+		;;
+	vax)
+		basic_machine=vax-dec
+		;;
+	pdp10)
+		# there are many clones, so DEC is not a safe bet
+		basic_machine=pdp10-unknown
+		;;
+	pdp11)
+		basic_machine=pdp11-dec
+		;;
+	we32k)
+		basic_machine=we32k-att
+		;;
+	sh3 | sh4 | sh[34]eb | sh[1234]le | sh[23]ele)
+		basic_machine=sh-unknown
+		;;
+	sh64)
+		basic_machine=sh64-unknown
+		;;
+	sparc | sparcv9 | sparcv9b)
+		basic_machine=sparc-sun
+		;;
+	cydra)
+		basic_machine=cydra-cydrome
+		;;
+	orion)
+		basic_machine=orion-highlevel
+		;;
+	orion105)
+		basic_machine=clipper-highlevel
+		;;
+	mac | mpw | mac-mpw)
+		basic_machine=m68k-apple
+		;;
+	pmac | pmac-mpw)
+		basic_machine=powerpc-apple
+		;;
+	*-unknown)
+		# Make sure to match an already-canonicalized machine name.
+		;;
+	*)
+		echo Invalid configuration \`$1\': machine \`$basic_machine\' not recognized 1>&2
+		exit 1
+		;;
+esac
+
+# Here we canonicalize certain aliases for manufacturers.
+case $basic_machine in
+	*-digital*)
+		basic_machine=`echo $basic_machine | sed 's/digital.*/dec/'`
+		;;
+	*-commodore*)
+		basic_machine=`echo $basic_machine | sed 's/commodore.*/cbm/'`
+		;;
+	*)
+		;;
+esac
+
+# Decode manufacturer-specific aliases for certain operating systems.
+
+if [ x"$os" != x"" ]
+then
+case $os in
+        # First match some system type aliases
+        # that might get confused with valid system types.
+	# -solaris* is a basic system type, with this one exception.
+	-solaris1 | -solaris1.*)
+		os=`echo $os | sed -e 's|solaris1|sunos4|'`
+		;;
+	-solaris)
+		os=-solaris2
+		;;
+	-svr4*)
+		os=-sysv4
+		;;
+	-unixware*)
+		os=-sysv4.2uw
+		;;
+	-gnu/linux*)
+		os=`echo $os | sed -e 's|gnu/linux|linux-gnu|'`
+		;;
+	# First accept the basic system types.
+	# The portable systems comes first.
+	# Each alternative MUST END IN A *, to match a version number.
+	# -sysv* is not here because it comes later, after sysvr4.
+	-gnu* | -bsd* | -mach* | -minix* | -genix* | -ultrix* | -irix* \
+	      | -*vms* | -sco* | -esix* | -isc* | -aix* | -sunos | -sunos[34]*\
+	      | -hpux* | -unos* | -osf* | -luna* | -dgux* | -solaris* | -sym* \
+	      | -amigaos* | -amigados* | -msdos* | -newsos* | -unicos* | -aof* \
+	      | -aos* \
+	      | -nindy* | -vxsim* | -vxworks* | -ebmon* | -hms* | -mvs* \
+	      | -clix* | -riscos* | -uniplus* | -iris* | -rtu* | -xenix* \
+	      | -hiux* | -386bsd* | -netbsd* | -openbsd* | -freebsd* | -riscix* \
+	      | -lynxos* | -bosx* | -nextstep* | -cxux* | -aout* | -elf* | -oabi* \
+	      | -ptx* | -coff* | -ecoff* | -winnt* | -domain* | -vsta* \
+	      | -udi* | -eabi* | -lites* | -ieee* | -go32* | -aux* \
+	      | -chorusos* | -chorusrdb* \
+	      | -cygwin* | -pe* | -psos* | -moss* | -proelf* | -rtems* \
+	      | -mingw32* | -linux-gnu* | -uxpv* | -beos* | -mpeix* | -udk* \
+	      | -interix* | -uwin* | -mks* | -rhapsody* | -darwin* | -opened* \
+	      | -openstep* | -oskit* | -conix* | -pw32* | -nonstopux* \
+	      | -storm-chaos* | -tops10* | -tenex* | -tops20* | -its* \
+	      | -os2* | -vos* | -palmos* | -uclinux* | -nucleus* \
+	      | -morphos* | -superux* | -rtmk* | -rtmk-nova* | -windiss* \
+	      | -powermax* | -dnix* | -nx6 | -nx7 | -sei*)
+	# Remember, each alternative MUST END IN *, to match a version number.
+		;;
+	-qnx*)
+		case $basic_machine in
+		    x86-* | i*86-*)
+			;;
+		    *)
+			os=-nto$os
+			;;
+		esac
+		;;
+	-nto-qnx*)
+		;;
+	-nto*)
+		os=`echo $os | sed -e 's|nto|nto-qnx|'`
+		;;
+	-sim | -es1800* | -hms* | -xray | -os68k* | -none* | -v88r* \
+	      | -windows* | -osx | -abug | -netware* | -os9* | -beos* \
+	      | -macos* | -mpw* | -magic* | -mmixware* | -mon960* | -lnews*)
+		;;
+	-mac*)
+		os=`echo $os | sed -e 's|mac|macos|'`
+		;;
+	-linux*)
+		os=`echo $os | sed -e 's|linux|linux-gnu|'`
+		;;
+	-sunos5*)
+		os=`echo $os | sed -e 's|sunos5|solaris2|'`
+		;;
+	-sunos6*)
+		os=`echo $os | sed -e 's|sunos6|solaris3|'`
+		;;
+	-opened*)
+		os=-openedition
+		;;
+	-wince*)
+		os=-wince
+		;;
+	-osfrose*)
+		os=-osfrose
+		;;
+	-osf*)
+		os=-osf
+		;;
+	-utek*)
+		os=-bsd
+		;;
+	-dynix*)
+		os=-bsd
+		;;
+	-acis*)
+		os=-aos
+		;;
+	-atheos*)
+		os=-atheos
+		;;
+	-386bsd)
+		os=-bsd
+		;;
+	-ctix* | -uts*)
+		os=-sysv
+		;;
+	-nova*)
+		os=-rtmk-nova
+		;;
+	-ns2 )
+		os=-nextstep2
+		;;
+	-nsk*)
+		os=-nsk
+		;;
+	# Preserve the version number of sinix5.
+	-sinix5.*)
+		os=`echo $os | sed -e 's|sinix|sysv|'`
+		;;
+	-sinix*)
+		os=-sysv4
+		;;
+	-triton*)
+		os=-sysv3
+		;;
+	-oss*)
+		os=-sysv3
+		;;
+	-svr4)
+		os=-sysv4
+		;;
+	-svr3)
+		os=-sysv3
+		;;
+	-sysvr4)
+		os=-sysv4
+		;;
+	# This must come after -sysvr4.
+	-sysv*)
+		;;
+	-ose*)
+		os=-ose
+		;;
+	-es1800*)
+		os=-ose
+		;;
+	-xenix)
+		os=-xenix
+		;;
+	-*mint | -mint[0-9]* | -*MiNT | -MiNT[0-9]*)
+		os=-mint
+		;;
+	-aros*)
+		os=-aros
+		;;
+	-kaos*)
+		os=-kaos
+		;;
+	-none)
+		;;
+	*)
+		# Get rid of the `-' at the beginning of $os.
+		os=`echo $os | sed 's/[^-]*-//'`
+		echo Invalid configuration \`$1\': system \`$os\' not recognized 1>&2
+		exit 1
+		;;
+esac
+else
+
+# Here we handle the default operating systems that come with various machines.
+# The value should be what the vendor currently ships out the door with their
+# machine or put another way, the most popular os provided with the machine.
+
+# Note that if you're going to try to match "-MANUFACTURER" here (say,
+# "-sun"), then you have to tell the case statement up towards the top
+# that MANUFACTURER isn't an operating system.  Otherwise, code above
+# will signal an error saying that MANUFACTURER isn't an operating
+# system, and we'll never get to this point.
+
+case $basic_machine in
+	*-acorn)
+		os=-riscix1.2
+		;;
+	arm*-rebel)
+		os=-linux
+		;;
+	arm*-semi)
+		os=-aout
+		;;
+    c4x-* | tic4x-*)
+        os=-coff
+        ;;
+	# This must come before the *-dec entry.
+	pdp10-*)
+		os=-tops20
+		;;
+	pdp11-*)
+		os=-none
+		;;
+	*-dec | vax-*)
+		os=-ultrix4.2
+		;;
+	m68*-apollo)
+		os=-domain
+		;;
+	i386-sun)
+		os=-sunos4.0.2
+		;;
+	m68000-sun)
+		os=-sunos3
+		# This also exists in the configure program, but was not the
+		# default.
+		# os=-sunos4
+		;;
+	m68*-cisco)
+		os=-aout
+		;;
+	mips*-cisco)
+		os=-elf
+		;;
+	mips*-*)
+		os=-elf
+		;;
+	or32-*)
+		os=-coff
+		;;
+	*-tti)	# must be before sparc entry or we get the wrong os.
+		os=-sysv3
+		;;
+	sparc-* | *-sun)
+		os=-sunos4.1.1
+		;;
+	*-be)
+		os=-beos
+		;;
+	*-ibm)
+		os=-aix
+		;;
+	*-wec)
+		os=-proelf
+		;;
+	*-winbond)
+		os=-proelf
+		;;
+	*-oki)
+		os=-proelf
+		;;
+	*-hp)
+		os=-hpux
+		;;
+	*-hitachi)
+		os=-hiux
+		;;
+	i860-* | *-att | *-ncr | *-altos | *-motorola | *-convergent)
+		os=-sysv
+		;;
+	*-cbm)
+		os=-amigaos
+		;;
+	*-dg)
+		os=-dgux
+		;;
+	*-dolphin)
+		os=-sysv3
+		;;
+	m68k-ccur)
+		os=-rtu
+		;;
+	m88k-omron*)
+		os=-luna
+		;;
+	*-next )
+		os=-nextstep
+		;;
+	*-sequent)
+		os=-ptx
+		;;
+	*-crds)
+		os=-unos
+		;;
+	*-ns)
+		os=-genix
+		;;
+	i370-*)
+		os=-mvs
+		;;
+	*-next)
+		os=-nextstep3
+		;;
+	*-gould)
+		os=-sysv
+		;;
+	*-highlevel)
+		os=-bsd
+		;;
+	*-encore)
+		os=-bsd
+		;;
+	*-sgi)
+		os=-irix
+		;;
+	*-siemens)
+		os=-sysv4
+		;;
+	*-masscomp)
+		os=-rtu
+		;;
+	f30[01]-fujitsu | f700-fujitsu)
+		os=-uxpv
+		;;
+	*-rom68k)
+		os=-coff
+		;;
+	*-*bug)
+		os=-coff
+		;;
+	*-apple)
+		os=-macos
+		;;
+	*-atari*)
+		os=-mint
+		;;
+	*)
+		os=-none
+		;;
+esac
+fi
+
+# Here we handle the case where we know the os, and the CPU type, but not the
+# manufacturer.  We pick the logical manufacturer.
+vendor=unknown
+case $basic_machine in
+	*-unknown)
+		case $os in
+			-riscix*)
+				vendor=acorn
+				;;
+			-sunos*)
+				vendor=sun
+				;;
+			-aix*)
+				vendor=ibm
+				;;
+			-beos*)
+				vendor=be
+				;;
+			-hpux*)
+				vendor=hp
+				;;
+			-mpeix*)
+				vendor=hp
+				;;
+			-hiux*)
+				vendor=hitachi
+				;;
+			-unos*)
+				vendor=crds
+				;;
+			-dgux*)
+				vendor=dg
+				;;
+			-luna*)
+				vendor=omron
+				;;
+			-genix*)
+				vendor=ns
+				;;
+			-mvs* | -opened*)
+				vendor=ibm
+				;;
+			-ptx*)
+				vendor=sequent
+				;;
+			-vxsim* | -vxworks* | -windiss*)
+				vendor=wrs
+				;;
+			-aux*)
+				vendor=apple
+				;;
+			-hms*)
+				vendor=hitachi
+				;;
+			-mpw* | -macos*)
+				vendor=apple
+				;;
+			-*mint | -mint[0-9]* | -*MiNT | -MiNT[0-9]*)
+				vendor=atari
+				;;
+			-vos*)
+				vendor=stratus
+				;;
+		esac
+		basic_machine=`echo $basic_machine | sed "s/unknown/$vendor/"`
+		;;
+esac
+
+echo $basic_machine$os
+exit 0
+
+# Local variables:
+# eval: (add-hook 'write-file-hooks 'time-stamp)
+# time-stamp-start: "timestamp='"
+# time-stamp-format: "%:y-%02m-%02d"
+# time-stamp-end: "'"
+# End:
/cvs/dm/dmraid/autoconf/install-sh,v  -->  standard output
revision 1.1
--- dmraid/autoconf/install-sh
+++ -	2008-02-22 16:57:38.022902000 +0000
@@ -0,0 +1,286 @@
+#!/bin/sh
+# install - install a program, script, or datafile
+
+scriptversion=2003-06-13.21
+
+# This originates from X11R5 (mit/util/scripts/install.sh), which was
+# later released in X11R6 (xc/config/util/install.sh) with the
+# following copyright and license.
+#
+# Copyright (C) 1994 X Consortium
+#
+# Permission is hereby granted, free of charge, to any person obtaining a copy
+# of this software and associated documentation files (the "Software"), to
+# deal in the Software without restriction, including without limitation the
+# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
+# sell copies of the Software, and to permit persons to whom the Software is
+# furnished to do so, subject to the following conditions:
+#
+# The above copyright notice and this permission notice shall be included in
+# all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
+# X CONSORTIUM BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
+# AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNEC-
+# TION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+# Except as contained in this notice, the name of the X Consortium shall not
+# be used in advertising or otherwise to promote the sale, use or other deal-
+# ings in this Software without prior written authorization from the X Consor-
+# tium.
+#
+#
+# FSF changes to this file are in the public domain.
+#
+# Calling this script install-sh is preferred over install.sh, to prevent
+# `make' implicit rules from creating a file called install from it
+# when there is no Makefile.
+#
+# This script is compatible with the BSD install script, but was written
+# from scratch.  It can only install one file at a time, a restriction
+# shared with many OS's install programs.
+
+# set DOITPROG to echo to test this script
+
+# Don't use :- since 4.3BSD and earlier shells don't like it.
+doit="${DOITPROG-}"
+
+# put in absolute paths if you don't have them in your path; or use env. vars.
+
+mvprog="${MVPROG-mv}"
+cpprog="${CPPROG-cp}"
+chmodprog="${CHMODPROG-chmod}"
+chownprog="${CHOWNPROG-chown}"
+chgrpprog="${CHGRPPROG-chgrp}"
+stripprog="${STRIPPROG-strip}"
+rmprog="${RMPROG-rm}"
+mkdirprog="${MKDIRPROG-mkdir}"
+
+transformbasename=
+transform_arg=
+instcmd="$mvprog"
+chmodcmd="$chmodprog 0755"
+chowncmd=
+chgrpcmd=
+stripcmd=
+rmcmd="$rmprog -f"
+mvcmd="$mvprog"
+src=
+dst=
+dir_arg=
+
+usage="Usage: $0 [OPTION]... SRCFILE DSTFILE
+   or: $0 -d DIR1 DIR2...
+
+In the first form, install SRCFILE to DSTFILE, removing SRCFILE by default.
+In the second, create the directory path DIR.
+
+Options:
+-b=TRANSFORMBASENAME
+-c         copy source (using $cpprog) instead of moving (using $mvprog).
+-d         create directories instead of installing files.
+-g GROUP   $chgrp installed files to GROUP.
+-m MODE    $chmod installed files to MODE.
+-o USER    $chown installed files to USER.
+-s         strip installed files (using $stripprog).
+-t=TRANSFORM
+--help     display this help and exit.
+--version  display version info and exit.
+
+Environment variables override the default commands:
+  CHGRPPROG CHMODPROG CHOWNPROG CPPROG MKDIRPROG MVPROG RMPROG STRIPPROG
+"
+
+while test -n "$1"; do
+  case $1 in
+    -b=*) transformbasename=`echo $1 | sed 's/-b=//'`
+        shift
+        continue;;
+
+    -c) instcmd=$cpprog
+        shift
+        continue;;
+
+    -d) dir_arg=true
+        shift
+        continue;;
+
+    -g) chgrpcmd="$chgrpprog $2"
+        shift
+        shift
+        continue;;
+
+    --help) echo "$usage"; exit 0;;
+
+    -m) chmodcmd="$chmodprog $2"
+        shift
+        shift
+        continue;;
+
+    -o) chowncmd="$chownprog $2"
+        shift
+        shift
+        continue;;
+
+    -s) stripcmd=$stripprog
+        shift
+        continue;;
+
+    -t=*) transformarg=`echo $1 | sed 's/-t=//'`
+        shift
+        continue;;
+
+    --version) echo "$0 $scriptversion"; exit 0;;
+
+    *)  if test -z "$src"; then
+          src=$1
+        else
+          # this colon is to work around a 386BSD /bin/sh bug
+          :
+          dst=$1
+        fi
+        shift
+        continue;;
+  esac
+done
+
+if test -z "$src"; then
+  echo "$0: no input file specified." >&2
+  exit 1
+fi
+
+if test -n "$dir_arg"; then
+  dst=$src
+  src=
+
+  if test -d "$dst"; then
+    instcmd=:
+    chmodcmd=
+  else
+    instcmd=$mkdirprog
+  fi
+else
+  # Waiting for this to be detected by the "$instcmd $src $dsttmp" command
+  # might cause directories to be created, which would be especially bad
+  # if $src (and thus $dsttmp) contains '*'.
+  if test ! -f "$src" && test ! -d "$src"; then
+    echo "$0: $src does not exist." >&2
+    exit 1
+  fi
+
+  if test -z "$dst"; then
+    echo "$0: no destination specified." >&2
+    exit 1
+  fi
+
+  # If destination is a directory, append the input filename; won't work
+  # if double slashes aren't ignored.
+  if test -d "$dst"; then
+    dst=$dst/`basename "$src"`
+  fi
+fi
+
+## this sed command emulates the dirname command
+dstdir=`echo "$dst" | sed -e 's,[^/]*$,,;s,/$,,;s,^$,.,'`
+
+# Make sure that the destination directory exists.
+# (this part is taken from Noah Friedman's mkinstalldirs script.)
+
+# Skip lots of stat calls in the usual case.
+if test ! -d "$dstdir"; then
+  defaultIFS='
+	'
+  IFS="${IFS-$defaultIFS}"
+
+  oIFS=$IFS
+  # Some sh's can't handle IFS=/ for some reason.
+  IFS='%'
+  set - `echo "$dstdir" | sed -e 's@/@%@g' -e 's@^%@/@'`
+  IFS=$oIFS
+
+  pathcomp=
+
+  while test $# -ne 0 ; do
+    pathcomp=$pathcomp$1
+    shift
+    test -d "$pathcomp" || $mkdirprog "$pathcomp"
+    pathcomp=$pathcomp/
+  done
+fi
+
+if test -n "$dir_arg"; then
+  $doit $instcmd "$dst" \
+    && { test -z "$chowncmd" || $doit $chowncmd "$dst"; } \
+    && { test -z "$chgrpcmd" || $doit $chgrpcmd "$dst"; } \
+    && { test -z "$stripcmd" || $doit $stripcmd "$dst"; } \
+    && { test -z "$chmodcmd" || $doit $chmodcmd "$dst"; }
+
+else
+  # If we're going to rename the final executable, determine the name now.
+  if test -z "$transformarg"; then
+    dstfile=`basename "$dst"`
+  else
+    dstfile=`basename "$dst" $transformbasename \
+             | sed $transformarg`$transformbasename
+  fi
+
+  # don't allow the sed command to completely eliminate the filename.
+  test -z "$dstfile" && dstfile=`basename "$dst"`
+
+  # Make a couple of temp file names in the proper directory.
+  dsttmp=$dstdir/_inst.$$_
+  rmtmp=$dstdir/_rm.$$_
+
+  # Trap to clean up those temp files at exit.
+  trap 'status=$?; rm -f "$dsttmp" "$rmtmp" && exit $status' 0
+  trap '(exit $?); exit' 1 2 13 15
+
+  # Move or copy the file name to the temp name
+  $doit $instcmd "$src" "$dsttmp" &&
+
+  # and set any options; do chmod last to preserve setuid bits.
+  #
+  # If any of these fail, we abort the whole thing.  If we want to
+  # ignore errors from any of these, just make sure not to ignore
+  # errors from the above "$doit $instcmd $src $dsttmp" command.
+  #
+  { test -z "$chowncmd" || $doit $chowncmd "$dsttmp"; } \
+    && { test -z "$chgrpcmd" || $doit $chgrpcmd "$dsttmp"; } \
+    && { test -z "$stripcmd" || $doit $stripcmd "$dsttmp"; } \
+    && { test -z "$chmodcmd" || $doit $chmodcmd "$dsttmp"; } &&
+
+  # Now remove or move aside any old file at destination location.  We
+  # try this two ways since rm can't unlink itself on some systems and
+  # the destination file might be busy for other reasons.  In this case,
+  # the final cleanup might fail but the new file should still install
+  # successfully.
+  {
+    if test -f "$dstdir/$dstfile"; then
+      $doit $rmcmd -f "$dstdir/$dstfile" 2>/dev/null \
+      || $doit $mvcmd -f "$dstdir/$dstfile" "$rmtmp" 2>/dev/null \
+      || {
+	  echo "$0: cannot unlink or rename $dstdir/$dstfile" >&2
+	  (exit 1); exit
+      }
+    else
+      :
+    fi
+  } &&
+
+  # Now rename the file to the real destination.
+  $doit $mvcmd "$dsttmp" "$dstdir/$dstfile"
+fi &&
+
+# The final little trick to "correctly" pass the exit status to the exit trap.
+{
+  (exit 0); exit
+}
+
+# Local variables:
+# eval: (add-hook 'write-file-hooks 'time-stamp)
+# time-stamp-start: "scriptversion="
+# time-stamp-format: "%:y-%02m-%02d.%02H"
+# time-stamp-end: "$"
+# End:
/cvs/dm/dmraid/doc/dmraid_design.txt,v  -->  standard output
revision 1.1
--- dmraid/doc/dmraid_design.txt
+++ -	2008-02-22 16:57:38.128560000 +0000
@@ -0,0 +1,283 @@
+
+dmraid tool design document v1.0.0-rc5f		Heinz Mauelshagen 2004.11.05
+----------------------------------------------------------------------------
+
+The dmraid tool supports RAID devices (RDs) such as ATARAID with
+device-mapper (dm) in Linux 2.6 avoiding the need to install a
+vendor specific (binary) driver to access them.
+
+It supports multiple on-disk RAID metadata formats and is open for
+extension with new ones.
+
+Initially dmraid aims to support activation of RDs and doesn't support
+*updates* of the ondisk metadata (eg, to record disk failures)
+It can optionally erase on disk metadata in case of multiple format
+signatures on a device.
+
+See future enhancements at the end.
+
+
+Functional requirements:
+------------------------
+
+1. dmraid must be able to read multiple vendor specific ondisk
+   RAID metadata formats:
+
+   o ATARAID
+     - Highpoint 37x/45x
+     - Intel Software RAID
+     - LSI Logic MegaRaid
+     - Promise FastTrak
+     - Silicon Image Medley
+
+2. dmraid shall be open to future extensions by other ondisk RAID formats,
+   eg. SNIA DDF
+   (http://www.snia.org/tech_activities/ddftwg/DDFTrial-UseDraft_0_45.pdf)
+
+3. dmraid shall generate the necessary dm table(s) defining
+   the needed mappings to address the RAID data.
+
+4. Device discovery, activation, deactivation and
+   property display shall be supported.
+
+5. Spanning of disks, RAID0, RAID1 and RAID10 shall be supported
+   (in order to be able to support SNIA DDF, higher raid levels need
+    implementing in form of respective dm targets; eg, RAID5);
+   Some vendors do have support for RAID5 already which is outside the scope
+   of dmraid because of the lag of a RAID5 target in device-mapper!
+
+6. Activation of MSDOS partitions in RAID sets shall be supported.
+
+
+Feature set definition:
+-----------------------
+
+Feature set summarizes as: Discover, Activate, Deactivate, Display.
+
+
+o Discover (1-n RD)
+
+  1 scan active disk devices identifying RD.
+
+  2 try to find an RD signature and if recognized,
+    add the device to the list of RDs found.
+
+  3 Abstract the metadata describing the RD layout and translate the vendor
+    specific format into it.
+
+
+o Activate (1-n RD)
+
+  1 group devices into abstracted RAID sets (RS) conforming to
+    their respective layout (SPAN, RAID0, RAID1, RAID10).
+  
+  2 generate dm mapping tables for a/those RS(s) derived from the abstracted
+    information about RS(s) and RD(s).
+  
+  3 create multiple/a dm device(s) for each RS to activate and 
+    load the generated table(s) into the device.
+
+  4 Recusively discover and activate MSDOS partitions in active RAID sets.
+
+
+o Deactivate (1-n RD)
+
+  1 remove the dm device(s) making up an RS; can be a stacked hierachy of
+    devices (eg, RAID10: RAID1 on top of n RAID0 devices).
+
+
+o Display (1-n RD)
+
+  1 display RS and RD properties
+    (eg, display information kept with RS(s) such as size and type)
+
+
+
+Technical specification:
+------------------------
+
+All functions returning int or a pointer return 0/NULL on failure.
+
+o RAID metadata format handler
+
+  Tool calls the following function to register a vendor specific
+  format handler; in case of success, a new instance with methods is
+  accessible to the high level metadata handling functions (see below):
+
+  - int register_format_handler(struct lib_context *lc,
+				struct dmraid_format *fmt);
+
+    x returns !0 on successfull format handler registration
+
+  - Format handler methods:
+
+    x struct raid_dev *(read)(struct lib_context *lc, struct dev_info* di);
+
+      - returns 'struct raid_dev *' describing the RD (eg, offset, length)
+
+    x int (*write)(struct lib_context *lc, struct raid_dev* rd, int erase);
+
+      - returns != 0 successfully written the vendor specific metadata
+        back to the respective RD rd (optionally erasing the
+	metadata area(s) in case erase != 0)
+
+    x struct raid_set (*group)(struct lib_context *lc,
+				 struct raid_dev *raid_dev)
+
+      - returns pointer to RAID set structure on success
+
+    x int (*check)(struct lib_context *lc, struct raid_set *raid_set)
+
+      - returns != 0 in case raid set is consistent
+
+    x void (*log)(struct lib_context *lc, struct raid_dev *rd)
+
+      - display metadata in native format (ie. all vendor specific
+	metadata fields) for RD rd
+
+
+o Discover
+
+  1 retrieve block device information from sysfs for all disk
+    devices by scanning /SYSFS_MOUNTPOINT/block/[sh]d*;
+    keep information about the device path and size which is the base
+    to find the RAID signature on the device in a linked list
+    of type 'struct dev_info *'.
+
+  2 walk the list and try to read RD metadata signature off the device
+    trying vendor specific read methods (eg, Highpoint...) in turn;
+    library exposes interface to register format handlers for vendor
+    specific RAID formats in order to be open for future extensions
+    (see register_format_handler() above).
+
+    Tool calls the following high level function which hides
+    the iteration through all the registered format handler methods:
+
+    x void discover_devices(struct lib_context *lc)
+
+      - returns != 0 in case it discovered all supported disks (IDE, SCSI)
+	and added them to a list
+
+    x void discover_raid_devices(struct lib_context *lc,
+				 char **fmt_names,
+				 char **devices);
+
+      - discovers all suported RDs using the list filled by
+ 	discover_devices() and adds them to a list
+
+    x void discover_partitions(struct lib_context *lc);
+
+      - discovers all MSDOS partitions in active RAID sets and builds
+	RAID sets with one linear device hanging off for activation.
+
+    x int count_devices(struct lib_context *lc, enum dev_type type);
+
+      - returns number of devices found by discover_devices() and
+	discover_raid_devices() of specified type (eg, DEVICE, RAID, ...)
+
+    x int perform(struct lib_context *lc, char **argv);
+
+      - returns != 0 on success performing various actions
+	(ie. activation/deacivation of RAID sets, displaying properties, ...)
+
+o Activate 1
+
+    Tool calls the following high level function which hide
+    the details of the RAID set assembly:
+
+    x int group_set(struct lib_context *lc, char *name);
+
+      - returns != 0 on successfully grouping an RS/RSs
+
+o Activate 2+3
+
+  - don't activate non-RAID1 devices which have an invalid set check result
+    and display an error
+  - create the ASCII dm mapping table by iterating through the list
+    of RD in a particular set, retrieving the layout (SPAN, ...)
+    the device path, the offset into the device and the length to map
+    and the stripe size in case of RAID
+  - create a unique device_name
+  - call device-mapper library to create the mapped device and load
+    the mapping table using this function:
+
+    x int activate_set(struct lib_context *lc, void *rs)
+
+      - returns != 0 in case of successfull RAID set activation
+
+o Activate 4
+  - activate MSDOS partitioned RAID sets as in 2+3
+
+
+o Deactivate
+
+  - check if a partitioned RAID set is active and deactivate it first
+
+  - check if the RAID set is active and call device-mapper library to
+    remove the mapped device (recursively in case of a mapped-device hierarchy)
+    using this function:
+
+    x int deactivate_set(struct lib_context *lc, void *rs)
+
+      - returns != 0 in case of successfull RAID set deactivation
+
+
+o Display
+
+  - list all block devices found
+  - list all (in)active RD
+  - display properties of a particular/all RD devices
+    (eg, members of the set by block device name and offset/length mapped
+     to those...)
+
+    x void display_devices(struct lib_context *lc, enum dev_type type);
+
+      - display devices of dev_type 'type' (eg, RAID, DEVICE, ...)
+
+    x void display_set(struct lib_context *lc, void *rs,
+		       enum active_type type, int top);
+
+      - display RS of active_type 'type' (ACTIVE, INACTIVE, ALL)
+
+
+Code directory tree:
+--------------------
+
+dmraid ---/doc
+	+-/include
+	+-/lib ---/activate
+	|	|-/datastruct
+	|	|-/device
+	|	|-/display
+	|	|-/format ---/ataraid
+	|	|	   +-/partition
+	|	|          +-/template
+	|	|-/locking
+	|	|-/log
+	|	|-/misc
+	|	|-/metadata
+	|	+-/mm
+	+-/man
+	+-/tools
+
+
+Future enhancements:
+--------------------
+
+o enhance write support to update ondisk metadata
+  - to restore metadata backups
+  - to record disk failures
+
+o support to log state (eg, sector failures) in standard/vendor ondisk logs;
+  needs above write support
+
+o status daemon to monitor RAID set sanity
+  (eg, disk failure, hot spare rebuild, ...) and
+  frontend with CLI
+
+
+Open questions:
+---------------
+
+o do we need to prioritize on device-mapper targets for higher RAID levels
+  (in particular we'ld need RAID3+5 to support some ATARAID formats) ?
/cvs/dm/dmraid/include/Makefile.in,v  -->  standard output
revision 1.1
--- dmraid/include/Makefile.in
+++ -	2008-02-22 16:57:38.215747000 +0000
@@ -0,0 +1,31 @@
+#
+# Copyright (C) 2005  Heinz Mauelshagen, Red Hat GmbH. All rights reserved.
+#
+# See file LICENSE at the top of this source tree for license information.
+#
+
+srcdir = @srcdir@
+top_srcdir = @top_srcdir@
+
+HEADERS=$(wildcard dmraid/*.h)
+
+all:
+
+include $(top_srcdir)/make.tmpl
+
+.PHONY: install_dmraid_headers remove_dmraid_headers
+
+install_dmraid_headers: $(HEADERS)
+	@echo "Installing $(HEADERS) in $(includedir)/dmraid"
+	mkdir -p $(includedir)/dmraid
+	$(INSTALL) $(STRIP) $(HEADERS) $(includedir)/dmraid
+
+install: install_dmraid_headers
+
+remove_dmraid_headers:
+	@echo "Removing $(HEADERS) from $(includedir)/dmraid"
+	rm -f $(includedir)/dmraid
+
+remove:        remove_dmraid_headers
+
+clean:
/cvs/dm/dmraid/include/dmraid/display.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/display.h
+++ -	2008-02-22 16:57:38.299803000 +0000
@@ -0,0 +1,30 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _DISPLAY_H_
+#define _DISPLAY_H_
+
+enum dev_type {
+	DEVICE = 0x01,	/* ALL devices */
+	RAID   = 0x02,	/* RAID devices */
+	NATIVE = 0x04,	/* Native metadata of RAID devices */
+	SET    = 0x08,	/* RAID sets */
+};
+
+enum active_type {
+	D_ALL      = 0x01,	/* All devices */
+	D_ACTIVE   = 0x02,	/* Active devices only */
+	D_INACTIVE = 0x04,	/* Inactive devices only */
+};
+
+extern void display_devices(struct lib_context *lc, enum dev_type type);
+extern void display_set(struct lib_context *lc, void *rs,
+			enum active_type active, int top);
+extern void display_table(struct lib_context *lc, char *rs_name, char *table);
+extern int list_formats(struct lib_context *lc, int arg);
+
+#endif
/cvs/dm/dmraid/include/dmraid/dmraid.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/dmraid.h
+++ -	2008-02-22 16:57:38.384392000 +0000
@@ -0,0 +1,108 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _DMRAID_H_
+#define _DMRAID_H_
+
+#include <stdint.h>
+#include <stdlib.h>
+#include <sys/types.h>
+
+/* FIXME: avoid more library internals. */
+#include <dmraid/lib_context.h>
+#include <dmraid/display.h>
+#include <dmraid/format.h>
+#include <dmraid/metadata.h>
+
+/*
+ * Library init/exit
+ */
+extern struct lib_context *libdmraid_init(int argc, char **argv);
+extern void libdmraid_exit(struct lib_context *lc);
+
+/*
+ * Retrieve version identifiers.
+ */
+extern int dm_version(struct lib_context *lc, char *version, size_t size);
+extern const char *libdmraid_date(struct lib_context *lc);
+extern const char *libdmraid_version(struct lib_context *lc);
+
+/*
+ * Dealing with formats.
+ */
+extern int check_valid_format(struct lib_context *lc, char *fmt);
+
+/*
+ * Dealing with devices.
+ */
+extern unsigned int count_devices(struct lib_context *lc, enum dev_type type);
+extern int discover_devices(struct lib_context *lc, char **devnodes);
+extern void discover_raid_devices(struct lib_context *lc, char **devices);
+extern void discover_partitions(struct lib_context *lc);
+
+/*
+ * Erase ondisk metadata.
+ */
+extern int erase_metadata(struct lib_context *lc);
+
+/*
+ * Dealing with RAID sets.
+ */
+extern const char *get_set_type(struct lib_context *lc, void *rs);
+extern const char *get_set_name(struct lib_context *lc, void *rs);
+extern int group_set(struct lib_context *lc, char *name);
+extern char *libdmraid_make_table(struct lib_context *lc, struct raid_set *rs);
+
+enum activate_type {
+	A_ACTIVATE,
+	A_DEACTIVATE,
+};
+
+extern void process_sets(struct lib_context *lc,
+			 int (*func)(struct lib_context *lc, void *rs, int arg),
+			 int arg, enum set_type type);
+extern int change_set(struct lib_context *lc, enum activate_type what,
+		      void *rs);
+
+/*
+ * Memory allocation
+ */
+#ifdef	DEBUG_MALLOC
+
+extern void *_dbg_malloc(size_t size, struct lib_context *lc,
+			 const char *who, unsigned int line);
+extern void *_dbg_realloc(void *ptr, size_t size, struct lib_context *lc,
+			  const char *who, unsigned int line);
+extern void *_dbg_strdup(void *ptr, struct lib_context *lc,
+			 const char *who, unsigned int line);
+extern void _dbg_free(void *ptr, struct lib_context *lc,
+		      const char *who, unsigned int line);
+
+#define	dbg_malloc(size)	_dbg_malloc((size), lc, __func__, __LINE__)
+#define	dbg_realloc(ptr, size)	_dbg_realloc((ptr), (size), lc, \
+					     __func__, __LINE__)
+#define	dbg_strdup(ptr)		_dbg_strdup((ptr), lc, __func__, __LINE__)
+#define	dbg_strndup(ptr, len)	_dbg_strndup((ptr), len, lc, __func__, __LINE__)
+#define	dbg_free(ptr)		_dbg_free((ptr), lc, __func__, __LINE__)
+
+#else
+
+extern void *_dbg_malloc(size_t size);
+extern void *_dbg_realloc(void *ptr, size_t size);
+extern void *_dbg_strdup(void *ptr);
+extern void *_dbg_strndup(void *ptr, size_t len);
+extern void _dbg_free(void *ptr);
+
+#define	dbg_malloc	_dbg_malloc
+#define	dbg_realloc	_dbg_realloc
+#define	dbg_strdup	_dbg_strdup
+#define	dbg_strndup	_dbg_strndup
+#define	dbg_free	_dbg_free
+
+#endif /* #ifdef DEBUG_MALLOC */
+
+#endif
/cvs/dm/dmraid/include/dmraid/format.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/format.h
+++ -	2008-02-22 16:57:38.468937000 +0000
@@ -0,0 +1,217 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_FORMAT_H_
+#define	_FORMAT_H_
+
+#ifdef FORMAT_HANDLER
+#undef FORMAT_HANDLER
+
+#include <sys/types.h>
+#include <dmraid/list.h>
+#include <dmraid/metadata.h>
+
+/* Metadata format handler types. */
+enum fmt_type {
+	FMT_RAID,
+	FMT_PARTITION,
+};
+
+/*
+ * Various structures for data whipped with the different event types.
+ */
+/* Event: I/O error */
+struct event_io {
+	struct raid_set *rs;	/* RAID set of I/O error. */
+	struct raid_dev *rd;	/* RAID device of I/O error. */
+	uint64_t sector;	/* Sector of the I/O error. */
+};
+
+/* Event: RAID device add/remove */
+enum rd_action {
+	rd_add,
+	rd_remove,
+};
+
+struct event_rd {
+	struct raid_set *rs;
+	struct raid_dev *rd;
+	enum rd_action	action;
+};
+
+/*
+ * List of event handler functions to call for the metadata format handler.
+ *
+ * Return 1 for event taken, RAID device write necessary.
+ * Return 0 for error and/or write unnecessary.
+ */
+struct event_handlers {
+	/* Handle IO error */
+	int (*io)(struct lib_context *lc, struct event_io *e_io);
+
+	/* Handle RAID device add/remove. */
+	int (*rd)(struct lib_context *lc, struct event_rd *e_rd);
+};
+
+/*
+ * Virtual interface definition of a metadata format handler.
+ */
+struct dmraid_format {
+	const char *name;	/* Format name */
+	const char *descr;	/* Format description */
+	const char *caps;	/* Capabilities (RAID levels supported) */
+	enum fmt_type format;	/* Format type (RAID, partition) */
+
+	/*
+	 * Read RAID metadata off a device and unify it.
+	 */
+	struct raid_dev* (*read)(struct lib_context *lc, struct dev_info* di);
+
+	/*
+	 * Write RAID metadata to a device deunifying it
+	 * or erase ondisk metadata if erase != 0.
+	 */
+	int (*write)(struct lib_context *lc, struct raid_dev* rd, int erase);
+
+	/*
+	 * Group a RAID device into a set.
+	 */
+	struct raid_set* (*group)(struct lib_context *lc, struct raid_dev *rd);
+
+	/*
+	 * Check consistency of the RAID set metadata.
+	 */
+	int (*check)(struct lib_context *lc, struct raid_set *rs);
+
+	/*
+	 * Event handlers (eg, I/O error).
+	 */
+	struct event_handlers *events;
+
+	/*
+	 * Display RAID disk metadata native.
+	 */
+	void (*log)(struct lib_context *lc, struct raid_dev *rd);
+};
+
+/* Chain of registered format handlers (needed for library context). */
+struct format_list {
+	struct list_head list;
+	struct dmraid_format *fmt;
+};
+
+int register_format_handlers(struct lib_context *lc);
+extern void unregister_format_handlers(struct lib_context *lc);
+
+
+/*
+ * Format core function used by (all) metadata format handlers.
+ */
+#define	NO_CHECK_RD	NULL
+extern int check_raid_set(struct lib_context *lc, struct raid_set *rs,
+			  unsigned int (*f_devices)(struct raid_dev *rd,
+						    void *context),
+			  void *f_devices_context,
+			  int (*f_check)(struct lib_context *lc,
+					 struct raid_set *rs,
+					 struct raid_dev *rd, void *context),
+			  void *f_check_context,
+			  const char *handler);
+extern int check_valid_format(struct lib_context *lc, char *fmt);
+extern int init_raid_set(struct lib_context *lc, struct raid_set *rs,
+			 struct raid_dev *rd, unsigned int stride,
+			 unsigned int type, const char *handler);
+extern const char **get_format_caps(struct lib_context *lc,
+				    struct dmraid_format *fmt);
+extern void free_format_caps(struct lib_context *lc, const char **caps);
+
+union read_info {
+	void *ptr;
+	uint32_t u32;
+	uint64_t u64;
+};
+
+struct raid_dev *read_raid_dev(
+	struct lib_context *lc,
+	struct dev_info *di,
+	void* (*f_read_metadata)(struct lib_context *lc, struct dev_info *di,
+				 size_t *size, uint64_t *offset,
+				 union read_info *info),
+	size_t size, uint64_t offset,
+	void (*f_to_cpu)(void *meta),
+	int (*f_is_meta)(struct lib_context *lc, struct dev_info *di,
+			 void *meta),
+	void (*f_file_metadata)(struct lib_context *lc, struct dev_info *di,
+				void *meta),
+	int (*f_setup_rd)(struct lib_context *lc, struct raid_dev *rd,
+			  struct dev_info *di, void *meta,
+			  union read_info *info),
+	const char *handler);
+
+extern void *alloc_meta_areas(struct lib_context *lc, struct raid_dev *rd,
+			      const char *who, unsigned int n);
+extern void *alloc_private(struct lib_context *lc, const char *who,
+			   size_t size);
+extern void *alloc_private_and_read(struct lib_context *lc, const char *who,
+				    size_t size, char *path, loff_t offset);
+extern struct raid_set *join_superset(
+	struct lib_context *lc,
+	char *(*f_name)(struct lib_context *lc, struct raid_dev *rd,
+			unsigned int subset),
+	void (*f_create)(struct raid_set *super, void *private),
+	int (*f_set_sort)(struct list_head *pos, struct list_head *new),
+	struct raid_set *rs, struct raid_dev *rd
+);
+extern int register_format_handler(struct lib_context *lc,
+				   struct dmraid_format *fmt);
+extern int write_metadata(struct lib_context *lc, const char *handler,
+			  struct raid_dev *rd, int meta_index, int erase);
+extern int log_zero_sectors(struct lib_context *lc, char *path,
+			    const char *handler);
+
+#define	to_disk	to_cpu
+
+#define struct_offset(s, member) ((unsigned short) &((struct s *) 0)->member)
+
+/* Print macros used in log methods. */
+
+/* Undefine this to avoid offsets in metadata logging. */
+#define	NATIVE_LOG_OFFSET
+#ifdef	NATIVE_LOG_OFFSET
+#define	P_FMT	"0x%03x "
+#define	P_OFF(x, basevar, y...) \
+	((unsigned long) &x - (unsigned long) basevar), y
+#else
+#define	P_FMT
+#define	P_OFF(x, basevar, y...)	y
+#endif
+
+#define	P(format, basevar, x, y...) \
+	do { log_print(lc, P_FMT format, P_OFF(x, basevar, y)); } while(0)
+
+#define	P2(format, basevar, i, x) \
+	do { P(format, basevar, x, i, x); } while(0)
+#define	DP(format, basevar, x) \
+	do { P(format, basevar, x, x); } while(0)
+
+/*
+ * RAID device, set and vendor metadata retrieval macros.
+ */
+#define	DEVS(rs)	(!list_empty(&((struct raid_set *) (rs))->devs))
+#define	SETS(rs)	(!list_empty(&((struct raid_set *) (rs))->sets))
+
+#define	META(rd, type)	((struct type*) ((struct raid_dev*) (rd))->meta_areas->area)
+#define	RD(pos)		(list_entry(pos, struct raid_dev, devs))
+#define	RS(pos)		(list_entry(pos, struct raid_set, list))
+#define RD_RS(rs)	(RD((((struct raid_set*) (rs))->devs.next)))
+#define RS_RS(rs)	((struct raid_set*) (rs)->sets.next)
+
+#define	HANDLER_LEN	sizeof(HANDLER)
+
+#endif /* ifdef FORMAT_HANDLER */
+
+#endif
/cvs/dm/dmraid/include/dmraid/lib_context.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/lib_context.h
+++ -	2008-02-22 16:57:38.559556000 +0000
@@ -0,0 +1,127 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _LIB_CONTEXT_H_
+#define _LIB_CONTEXT_H_
+
+#include <dmraid/list.h>
+#include <dmraid/locking.h>
+#include <dmraid/misc.h>
+
+enum lc_lists {
+	LC_FORMATS = 0,	/* Metadata format handlers. */
+	LC_DISK_INFOS,	/* Disks discovered. */
+	LC_RAID_DEVS,	/* Raid devices discovered. */
+	LC_RAID_SETS,	/* Raid sets grouped. */
+	/* Add new lists below here ! */
+	LC_LISTS_SIZE,	/* Must be the last enumerator. */
+};
+
+/* List access macros. */
+#define	LC_FMT(lc)	(lc_list((lc), LC_FORMATS))
+#define	LC_DI(lc)	(lc_list((lc), LC_DISK_INFOS))
+#define	LC_RD(lc)	(lc_list((lc), LC_RAID_DEVS))
+#define	LC_RS(lc)	(lc_list((lc), LC_RAID_SETS))
+
+enum lc_options {
+	LC_COLUMN = 0,
+	LC_DEBUG,
+	LC_DUMP,
+	LC_FORMAT,
+	LC_GROUP,
+	LC_SETS,
+	LC_TEST,
+	LC_VERBOSE,
+	LC_IGNORELOCKING,
+	LC_SEPARATOR,
+	LC_DEVICES,	  /* Add new options below this one ! */
+	LC_OPTIONS_SIZE,  /* Must be the last enumerator. */
+};
+
+/* Options access macros. */
+/* Return option counter. */
+#define	OPT_COLUMN(lc)		(lc_opt(lc, LC_COLUMN))
+#define	OPT_DEBUG(lc)		(lc_opt(lc, LC_DEBUG))
+#define	OPT_DEVICES(lc)		(lc_opt(lc, LC_DEVICES))
+#define	OPT_DUMP(lc)		(lc_opt(lc, LC_DUMP))
+#define	OPT_GROUP(lc)		(lc_opt(lc, LC_GROUP))
+#define	OPT_FORMAT(lc)		(lc_opt(lc, LC_FORMAT))
+#define	OPT_IGNORELOCKING(lc)	(lc_opt(lc, LC_IGNORELOCKING))
+#define	OPT_SEPARATOR(lc)	(lc_opt(lc, LC_SEPARATOR))
+#define	OPT_SETS(lc)		(lc_opt(lc, LC_SETS))
+#define	OPT_TEST(lc)		(lc_opt(lc, LC_TEST))
+#define	OPT_VERBOSE(lc)		(lc_opt(lc, LC_VERBOSE))
+
+/* Return option value. */
+#define	OPT_STR(lc, o)		(lc->options[o].arg.str)
+#define	OPT_STR_COLUMN(lc)	OPT_STR(lc, LC_COLUMN)
+#define	OPT_STR_FORMAT(lc)	OPT_STR(lc, LC_FORMAT)
+#define	OPT_STR_SEPARATOR(lc)	OPT_STR(lc, LC_SEPARATOR)
+
+struct lib_version {
+	const char *text;
+	const char *date;
+	struct {
+		unsigned int major;
+		unsigned int minor;
+		unsigned int sub_minor;
+		const char *suffix;
+	} v;
+};
+
+struct lib_options {
+	int opt;
+	union {
+		const char *str;
+		uint64_t u64;
+		uint64_t u32;
+	} arg;
+};
+
+struct lib_context {
+	struct lib_version version;
+	char *cmd;
+
+	/* Option counters used throughout the library. */
+	struct lib_options options[LC_OPTIONS_SIZE];
+
+	/*
+	 * Lists for:
+	 *
+	 *	o metadata format handlers the library supports
+	 * 	o block devices discovered
+	 * 	o RAID devices discovered
+	 * 	o RAID sets grouped
+	 */
+	struct list_head lists[LC_LISTS_SIZE];
+
+	char *locking_name;		/* Locking mechanism selector. */
+	struct	locking *lock;		/* Resource locking. */
+
+	mode_t	mode;			/* File/directrory create modes. */
+
+	struct {
+		const char *error;	/* For error mappings. */
+	} path;
+};
+
+extern struct lib_context *alloc_lib_context(char **argv);
+extern void free_lib_context(struct lib_context *lc);
+extern int lc_opt(struct lib_context *lc, enum lc_options o);
+const char *lc_opt_arg(struct lib_context *lc, enum lc_options o);
+const char *lc_stralloc_opt(struct lib_context *lc, enum lc_options o,
+			   char *arg);
+const char *lc_strcat_opt(struct lib_context *lc, enum lc_options o,
+			  char *arg, const char delim);
+extern int lc_inc_opt(struct lib_context *lc, int o);
+extern struct list_head *lc_list(struct lib_context *lc, int l);
+
+extern const char *libdmraid_date(struct lib_context *lc);
+extern const char *libdmraid_version(struct lib_context *lc);
+
+
+#endif
/cvs/dm/dmraid/include/dmraid/list.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/list.h
+++ -	2008-02-22 16:57:38.647160000 +0000
@@ -0,0 +1,70 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _LIST_H
+#define _LIST_H
+
+/*
+ * Double-linked list definitons and macros.
+ */
+
+struct list_head {
+	struct list_head *next;
+	struct list_head *prev;
+};
+
+#define INIT_LIST_HEAD(a)	do { (a)->next = (a)->prev = a; } while(0)
+
+#define list_empty(pos)	((pos)->next == pos)
+
+static inline void __list_add(struct list_head *new,
+			      struct list_head *prev,
+			      struct list_head *next)
+{
+	next->prev = new;
+	new->next = next;
+	new->prev = prev;
+	prev->next = new;
+}
+
+/* Add an element 'new' after position 'pos' to a list. */
+#define list_add(new, pos)	__list_add(new, pos, (pos)->next)
+
+/* Add an element 'new' before position 'pos' to a list. */
+#define list_add_tail(new, pos)	__list_add(new, (pos)->prev, pos)
+
+/* Delete an element 'pos' from the list. */
+#define	list_del(pos) { \
+	(pos)->next->prev = (pos)->prev; \
+	(pos)->prev->next = (pos)->next; \
+	(pos)->next = (pos)->prev = 0; \
+}
+
+/* Pointer to a struct 'type' derived from 'pos' and list_head* 'member'. */
+#define list_entry(pos, type, member) \
+	((type*) ((char*)pos - (unsigned long)(&((type*)0)->member)))
+
+/* Walk a list. */
+#define list_for_each(pos, head) \
+	for (pos = (head)->next; pos != head; pos = pos->next)
+
+/* Walk a list by entry. */
+#define list_for_each_entry(entry, head, member) \
+        for (entry = list_entry((head)->next, typeof(*entry), member); \
+             &entry->member != (head); \
+             entry = list_entry(entry->member.next, typeof(*entry), member))
+
+/*
+ * Walk a list using a temporary pointer,
+ * so that elements can be deleted safely.
+ */
+#define list_for_each_safe(pos, n, head) \
+        for (pos = (head)->next, n = pos->next; \
+             pos != (head); \
+             pos = n, n = pos->next)
+
+#endif
/cvs/dm/dmraid/include/dmraid/locking.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/locking.h
+++ -	2008-02-22 16:57:38.727484000 +0000
@@ -0,0 +1,29 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_LOCKING_H
+#define	_LOCKING_H
+
+/* I want to be prepared for finer grained locking... */
+struct resource {
+	char *name;
+};
+
+/* Locking abstraction. */
+struct lib_context;
+struct locking {
+	const char *name;
+	int (*lock)(struct lib_context *lc, struct resource *res);
+	void (*unlock)(struct lib_context *lc, struct resource *res);
+	void *private; /* Private context. */
+};
+
+extern int init_locking(struct lib_context *lc);
+extern int lock_resource(struct lib_context *lc, struct resource *res);
+extern void unlock_resource(struct lib_context *lc, struct resource *res);
+
+#endif
/cvs/dm/dmraid/include/dmraid/metadata.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/metadata.h
+++ -	2008-02-22 16:57:38.809297000 +0000
@@ -0,0 +1,254 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_META_H_
+#define	_META_H_
+
+#include <dmraid/list.h>
+#include <stdint.h>
+
+/*
+ * Unified RAID set types.
+ */
+enum type {
+	t_undef     = 0x01,
+	t_group     = 0x02, /* To group subsets (eg, Intel Software RAID). */
+	t_partition = 0x04, /* FIXME: remove in favour of kpartx ? */
+	t_spare     = 0x08,
+	t_linear    = 0x10,
+	t_raid0     = 0x20,
+	t_raid1     = 0x40,
+	/*
+	 * Higher RAID types below not supported (yet)
+	 * because of device-mapper constraints.
+	 */
+	t_raid4     = 0x80,
+	t_raid5_ls  = 0x100,
+	t_raid5_rs  = 0x200,
+	t_raid5_la  = 0x400,
+	t_raid5_ra  = 0x800,
+	t_raid6     = 0x1000,
+};
+
+/* Check macros for types. */
+#define	T_UNDEF(r)	((r)->type & t_undef)
+#define	T_GROUP(r)	((r)->type & t_group)
+#define	T_PARTITION(r)	((r)->type & t_partition)
+#define	T_SPARE(r)	((r)->type & t_spare)
+#define	T_LINEAR(r)	((r)->type & t_linear)
+#define	T_RAID0(r)	((r)->type & t_raid0)
+#define	T_RAID1(r)	((r)->type & t_raid1)
+#define	T_RAID4(r)	((r)->type & t_raid4)
+#define	T_RAID5(r)	(((r)->type & t_raid5_ls) || \
+			((r)->type & t_raid5_rs) || \
+			((r)->type & t_raid5_la) || \
+			((r)->type & t_raid5_ra))
+#define	T_RAID6(r)	((r)->type & t_raid6)
+
+
+/* Types for count_devs(). */
+enum count_type {
+	ct_all = 0,
+	ct_dev,
+	ct_spare,
+};
+
+/*
+ * Mapping struct for RAID type unification.
+ *
+ * Format handler allocates an array and inserts mappings
+ * from format specific types to the unified ones above.
+ */
+struct types {
+	unsigned int type; /* Must be long enough for vendor definition. */
+	enum type    unified_type;
+};
+
+/* RAID disk/set status. */
+enum status {
+	s_undef		= 0x01,
+	s_broken	= 0x02,	/* Completely broken (not accessible). */
+	s_inconsistent	= 0x04,	/* RAID disk/set inconsistent (needs
+				   synchronization or reconfiguration). */
+	/* FIXME: is s_nosync sufficient or do I need s_upgrade (eg, NVidia) */
+	s_nosync	= 0x08,	/* RAID disk/set *not* in sync
+				   (needs synchronization). */
+	s_ok		= 0x10,	/* Fully operational. */
+	s_setup		= 0x20,	/* Only during RAID setup transition. */
+};
+
+/* Check macros for states. */
+#define	S_UNDEF(status)		((status) & s_undef)
+#define	S_BROKEN(status)	((status) & s_broken)
+#define	S_INCONSISTENT(status)	((status) & s_inconsistent)
+#define	S_NOSYNC(status)	((status) & s_nosync)
+#define	S_OK(status)		((status) & s_ok)
+#define	S_SETUP(status)		((status) & s_setup)
+
+
+/* find_*() function enums */
+enum find {
+	FIND_TOP,	/* Search top level RAID sets only. */
+	FIND_ALL,	/* Decend all RAID set trees. */
+};
+
+/* Device information. */
+struct dev_info {
+	struct list_head list;	/* Global chain of discovered devices. */
+
+	char *path;	  /* Actual device node path. */
+	char *serial;	  /* ATA/SCSI serial number. */
+	uint64_t sectors; /* Device size. */
+};
+
+/* Metadata areas and size stored on a RAID device. */
+struct meta_areas {
+	uint64_t	offset;	/* on disk metadata offset in sectors. */
+	size_t		size;	/* on disk metadata size in bytes. */
+	void		*area;	/* pointer to format specific metadata. */
+};
+
+/*
+ * Abstracted RAID device.
+ *
+ * A RAID device is a member of a RAID set and can only
+ * exist at the lowest level of a RAID set stack (eg, for RAID10).
+ */
+struct raid_dev {
+	struct list_head list;		/* Global chain of RAID devices. */
+	struct list_head devs;		/* Chain of devices belonging to set. */
+
+	char		*name;		/* Metadata format handler generated
+					   name of set this device belongs to.*/
+
+	struct dev_info  *di;		/* Pointer to dev_info. */
+	struct dmraid_format *fmt;	/* Format handler for this device. */
+
+	enum status status;		/* Status of device. */
+	enum type type;			/* Type of device. */
+
+	uint64_t	offset;		/* Data offset on device. */
+	uint64_t	sectors;	/* Length of the segment to map. */
+
+	unsigned int	areas;		/* # of metadata areas on the device. */
+	struct meta_areas *meta_areas;	/* Dynamic array of metadata areas. */
+
+	/*
+	 * For format handler use (eg, to keep references between calls).
+	 * 
+	 * WARNING: non pointer members need to get zeroed before exit,
+	 *	    because the metadata layer frees private->ptr on cleanup.
+	 */
+	union {
+		void *ptr;
+		uint32_t n32;
+		uint64_t n64;
+	} private;
+};
+
+/*
+ * Abstracted (hierarchical) RAID set.
+ *
+ * Can be used to form a tree of subsets with arbitrary depths.
+ * Defines RAID attributes for the set as a whole (eg: RAID0, Status).
+ */
+enum flags {
+	f_maximize	= 0x01,	/* If set, maximize set capacity,
+				   if not set, limit to smallest device. */
+	f_partitions	= 0x02,	/* Set has partitions. */
+};
+
+#define	F_MAXIMIZE(rs)		((rs)->flags & f_maximize)
+#define	F_PARTITIONS(rs)	((rs)->flags & f_partitions)
+
+struct raid_set {
+	struct list_head list;		/* Chain of independent sets. */
+
+	/*
+	 * List of subsets (eg, RAID10) which make up RAID set stacks.
+	 *
+	 * Lowest level identified by list_empty() here.
+	 */
+	struct list_head sets;
+
+	/*
+	 * List of RAID devices making up a set.
+	 *
+	 * Higher RAID sets in a stack will have list_empty() here.
+	 *
+	 * Lowest level will hold device definitions
+	 * for arbitrary block devices here.
+	 */
+	struct list_head devs;
+	unsigned int total_devs;	/* The number of devices expected */
+	unsigned int found_devs;	/* The number of devices found */
+
+	char *name;			/* Name of the set. */
+
+	unsigned int stride;		/* Stride size. */
+	enum type type;			/* Unified raid type. */
+	enum flags flags;		/* Set flags. */
+	enum status status;		/* Status of set. */
+};
+
+extern struct dmraid_format *get_format(struct raid_set *rs);
+extern const char *get_type(struct lib_context *lc, enum type type);
+extern const char *get_dm_type(struct lib_context *lc, enum type type);
+extern const char *get_set_type(struct lib_context *lc, void *rs);
+extern const char *get_status(struct lib_context *lc, enum status status);
+extern uint64_t total_sectors(struct lib_context *lc, struct raid_set *rs);
+extern struct dev_info *alloc_dev_info(struct lib_context *lc, char *path);
+extern void free_dev_info(struct lib_context *lc, struct dev_info *di);
+extern struct raid_dev *alloc_raid_dev(struct lib_context *lc, const char *who);
+extern void free_raid_dev(struct lib_context *lc, struct raid_dev **rd);
+extern void list_add_sorted(struct lib_context *lc,
+		     struct list_head *to, struct list_head *new,
+		     int (*sort)(struct list_head *pos,
+				 struct list_head *new));
+extern struct raid_set *alloc_raid_set(struct lib_context *lc, const char *who);
+extern unsigned int count_sets(struct lib_context *lc, struct list_head *list);
+extern unsigned int count_devs(struct lib_context *lc, struct raid_set *rs,
+			enum count_type type);
+extern void free_raid_set(struct lib_context *lc, struct raid_set *rs);
+extern struct raid_set *find_set(struct lib_context *lc, struct list_head *list,
+			  const char *name, enum find where);
+extern struct raid_set *find_or_alloc_raid_set(struct lib_context *lc,
+					char *name, enum find where,
+					struct raid_dev *rd,
+					struct list_head *list,
+					void (*create) (struct raid_set *super,
+							void *private),
+					void *private);
+#define	NO_RD		NULL
+#define	NO_LIST		NULL
+#define	NO_CREATE	NULL
+#define	NO_CREATE_ARG	NULL
+extern const char *get_set_name(struct lib_context *lc, void *rs);
+extern int group_set(struct lib_context *lc, char *name);
+
+enum set_type {
+	SETS,
+	PARTITIONS,
+};
+
+extern void process_sets(struct lib_context *lc,
+		  int (*func)(struct lib_context *lc, void *rs, int arg),
+		  int arg, enum set_type type);
+extern int write_set(struct lib_context *lc, void *rs);
+extern int partitioned_set(struct lib_context *lc, void *rs);
+extern int base_partitioned_set(struct lib_context *lc, void *rs);
+extern void discover_raid_devices(struct lib_context *lc, char **devices);
+extern void discover_partitions(struct lib_context *lc);
+extern unsigned int count_devices(struct lib_context *lc, enum dev_type type);
+extern enum type rd_type(struct types *types, unsigned int type);
+extern void file_metadata(struct lib_context *lc, const char *handler,
+		   char *path, void *data, size_t size, uint64_t offset);
+extern void file_dev_size(struct lib_context *lc, const char *handler,
+		   struct dev_info *di);
+extern int erase_metadata(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/include/dmraid/misc.h,v  -->  standard output
revision 1.1
--- dmraid/include/dmraid/misc.h
+++ -	2008-02-22 16:57:38.894256000 +0000
@@ -0,0 +1,51 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _MISC_H_
+#define _MISC_H_
+
+extern struct lib_context *libdmraid_init(int argc, char **argv);
+extern void libdmraid_exit(struct lib_context *lc);
+
+extern void sysfs_workaround(struct lib_context *lc);
+extern void mk_alpha(struct lib_context *lc, char *str, size_t len);
+extern char *get_basename(struct lib_context *lc, char *str);
+extern char *get_dirname(struct lib_context *lc, char *str);
+extern char *remove_white_space(struct lib_context *lc, char *str, size_t len);
+extern char *remove_delimiter(char *ptr, char c);
+extern void add_delimiter(char **ptr, char c);
+
+extern int mk_dir(struct lib_context *lc, const char *dir);
+
+extern int read_file(struct lib_context *lc, const char *who, char *path,
+		     void *buffer, size_t size, loff_t offset);
+extern int write_file(struct lib_context *lc, const char *who, char *path,
+		      void *buffer, size_t size, loff_t offset);
+
+extern int yes_no_prompt(struct lib_context *lc, const char *prompt, ...);
+
+extern void free_string(struct lib_context *lc, char **string);
+extern int p_fmt(struct lib_context *lc, char **string, const char *fmt, ...);
+
+static inline uint64_t round_down(uint64_t what, unsigned int by)
+{
+	return what & ~((uint64_t) by - 1);
+}
+
+static inline uint64_t round_up(uint64_t what, unsigned int by)
+{
+	uint64_t t = round_down(what, by);
+
+	return t == what ? t : t + by;
+}
+
+static inline uint64_t div_up(uint64_t what, unsigned int by)
+{
+	return round_up(what, by) / by;
+}
+
+#endif
/cvs/dm/dmraid/lib/.export.sym,v  -->  standard output
revision 1.1
--- dmraid/lib/.export.sym
+++ -	2008-02-22 16:57:38.987669000 +0000
@@ -0,0 +1,51 @@
+Base {
+	global:
+		add_delimiter;
+		change_set;
+		check_valid_format;
+		collapse_delimiter;
+		count_devices;
+		count_devs;
+		count_sets;
+		_dbg_free;
+		_dbg_malloc;
+		_dbg_realloc;
+		_dbg_strdup;
+		discover_devices;
+		discover_partitions;
+		discover_raid_devices;
+		display_devices;
+		display_set;
+		dm_version;
+		erase_metadata;
+		find_set;
+		get_dm_type;
+		get_set_type;
+		get_set_name;
+		get_status;
+		get_type;
+		group_set;
+		init_locking;
+		lc_inc_opt;
+		lc_list;
+		lc_opt;
+		lc_stralloc_opt;
+		lc_strcat_opt;
+		libdmraid_exit;
+		libdmraid_init;
+		libdmraid_date;
+		libdmraid_version;
+		libdmraid_make_table;
+		list_formats;
+		lock_resource;
+		log_alloc_err;
+		plog;
+		process_sets;
+		remove_delimiter;
+		remove_white_space;
+		total_sectors;
+		unlock_resource;
+
+	local:
+		*;
+};
/cvs/dm/dmraid/lib/Makefile.in,v  -->  standard output
revision 1.1
--- dmraid/lib/Makefile.in
+++ -	2008-02-22 16:57:39.075357000 +0000
@@ -0,0 +1,89 @@
+#
+# Copyright (C) 2004-2005  Heinz Mauelshagen, Red Hat GmbH. All rights reserved.
+#
+# See file LICENSE at the top of this source tree for license information.
+#
+
+srcdir = @srcdir@
+top_srcdir = @top_srcdir@
+VPATH = @srcdir@
+
+SOURCES=\
+	activate/activate.c \
+	activate/devmapper.c \
+	device/ata.c \
+	device/scan.c \
+	device/scsi.c \
+	display/display.c \
+	format/format.c \
+	locking/locking.c \
+	log/log.c \
+	metadata/metadata.c \
+	misc/file.c \
+	misc/init.c \
+	misc/lib_context.c \
+	misc/misc.c \
+	misc/workaround.c \
+	mm/dbg_malloc.c \
+	format/ataraid/asr.c \
+	format/ataraid/hpt37x.c \
+	format/ataraid/hpt45x.c \
+	format/ataraid/isw.c \
+	format/ataraid/jm.c \
+	format/ataraid/lsi.c \
+	format/ataraid/nv.c \
+	format/ataraid/pdc.c \
+	format/ataraid/sil.c \
+	format/ataraid/via.c \
+	format/ataraid/asr.c \
+	format/partition/dos.c
+
+OBJECTS=$(SOURCES:%.c=%.o)
+
+LIB_STATIC=$(top_srcdir)/lib/libdmraid.a
+
+TARGETS=$(LIB_STATIC)
+INSTALL_TARGETS= $(LIB_STATIC)
+
+ifeq ("@KLIBC@", "no")
+  ifeq ("@LIB_SO@", "yes")
+    LIB_SHARED=$(top_srcdir)/lib/libdmraid.so
+    TARGETS += $(LIB_SHARED)
+    INSTALL_TARGETS += $(LIB_SHARED)
+  endif
+endif
+
+all:
+	@echo "$(TARGETS) $(INSTALL_TARGETS)"
+
+include $(top_srcdir)/make.tmpl
+
+.PHONY: install_dmraid_libs remove_dmraid_libs
+
+install_dmraid_libs: $(INSTALL_TARGETS)
+	@echo "Installing $(INSTALL_TARGETS) in $(libdir)"; \
+	mkdir -p $(libdir); \
+	for f in $(INSTALL_TARGETS); \
+	do \
+		n=$$(basename $${f}) ; \
+		if [[ "$$n" =~ '.so$$' ]]; then \
+			$(INSTALL) -m 555 $(STRIP) \
+				$$f $(libdir)/$${n}. at DMRAID_LIB_VERSION@; \
+			$(LN_S) -f $${n}. at DMRAID_LIB_VERSION@ $(libdir)/$${n}; \
+		else \
+			$(INSTALL) -m 555 $(STRIP) $$f $(libdir)/$${n}; \
+		fi \
+	done
+
+install: install_dmraid_libs
+
+remove_dmraid_libs:
+	@echo "Removing $(INSTALL_TARGETS) from $(libdir)"; \
+	for f in $(INSTALL_TARGETS); \
+	do \
+		n=$$(basename $${f}) ; \
+		rm -f $(libdir)/$${n}. at DMRAID_LIB_VERSION@; \
+		rm -f $(libdir)/$${n}; \
+	done
+
+remove:	remove_dmraid_libs
/cvs/dm/dmraid/lib/internal.h,v  -->  standard output
revision 1.1
--- dmraid/lib/internal.h
+++ -	2008-02-22 16:57:39.157284000 +0000
@@ -0,0 +1,55 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_INTERNAL_H_
+#define	_INTERNAL_H_
+
+#ifndef _GNU_SOURCE
+#define _GNU_SORUCE
+#endif
+
+#include <ctype.h>
+#include <fcntl.h>
+#include <inttypes.h>
+#include <limits.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <stdint.h>
+#include <dmraid/lib_context.h>
+#include <dmraid/list.h>
+#include <dmraid/locking.h>
+#include "log/log.h"
+#include "mm/dbg_malloc.h"
+#include <dmraid/misc.h>
+#include <dmraid/display.h>
+#include "device/dev-io.h"
+#define FORMAT_HANDLER
+#include <dmraid/format.h>
+#include <dmraid/metadata.h>
+#include "activate/activate.h"
+
+#ifndef	u_int16_t
+#define	u_int16_t	uint16_t
+#endif
+
+#ifndef	u_int32_t
+#define	u_int32_t	uint32_t
+#endif
+
+#ifndef	u_int64_t
+#define	u_int64_t	uint64_t
+#endif
+
+#define min(a, b) a < b ? a : b
+#define max(a, b) a > b ? a : b
+#define ARRAY_SIZE(a)   (sizeof(a) / sizeof(*a))
+#define ARRAY_END(a)   (a + ARRAY_SIZE(a))
+
+#endif
/cvs/dm/dmraid/lib/version.h,v  -->  standard output
revision 1.1
--- dmraid/lib/version.h
+++ -	2008-02-22 16:57:39.236706000 +0000
@@ -0,0 +1,12 @@
+#ifndef	DMRAID_LIB_VERSION
+
+#define	DMRAID_LIB_VERSION		"1.0.0.rc11"
+
+#define	DMRAID_LIB_MAJOR_VERSION	1
+#define	DMRAID_LIB_MINOR_VERSION	0
+#define	DMRAID_LIB_SUBMINOR_VERSION	0
+#define	DMRAID_LIB_VERSION_SUFFIX	"rc11"
+
+#define	DMRAID_LIB_DATE			"(2006.05.15)"
+
+#endif
/cvs/dm/dmraid/lib/activate/activate.c,v  -->  standard output
revision 1.1
--- dmraid/lib/activate/activate.c
+++ -	2008-02-22 16:57:39.319354000 +0000
@@ -0,0 +1,689 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Activate/Deactivate code for hierarchical RAID Sets.
+ */
+
+#include "internal.h"
+#include "devmapper.h"
+
+static int valid_rd(struct raid_dev *rd)
+{
+	return S_OK(rd->status) && !T_SPARE(rd);
+}
+
+static int valid_rs(struct raid_set *rs)
+{
+	return S_OK(rs->status) && !T_SPARE(rs);
+}
+
+/* Return rounded size in case of unbalanced mappings */
+static uint64_t maximize(struct raid_set *rs, uint64_t sectors,
+			 uint64_t last, uint64_t min)
+{
+	return sectors > min ? min(last, sectors) : last;
+}
+
+/* Find smallest set/disk larger than given minimum. */
+static uint64_t _smallest(struct lib_context *lc,
+			  struct raid_set *rs, uint64_t min)
+{
+	uint64_t ret = ~0;
+	struct raid_set *r;
+	struct raid_dev *rd;
+
+	list_for_each_entry(r, &rs->sets, list)
+		ret = maximize(r, total_sectors(lc, r), ret, min);
+
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (!T_SPARE(rd))
+			ret = maximize(rs, rd->sectors, ret, min);
+	}
+
+	return ret == (uint64_t) ~0 ? 0 : ret;
+}
+
+/*
+ * Definitions of mappings.
+ */
+
+/* Undefined/-supported mapping. */
+static int _dm_un(struct lib_context *lc, char **table,
+		  struct raid_set *rs, const char *what)
+{
+	LOG_ERR(lc, 0, "Un%sed RAID type %s[%u] on %s", what,
+		get_set_type(lc, rs), rs->type, rs->name);
+}
+
+static int dm_undef(struct lib_context *lc, char **table, struct raid_set *rs)
+{
+	return _dm_un(lc, table, rs, "defin");
+}
+
+static int dm_unsup(struct lib_context *lc, char **table, struct raid_set *rs)
+{
+	return _dm_un(lc, table, rs, "support");
+}
+
+
+/* "Spare mapping". */
+static int dm_spare(struct lib_context *lc, char **table, struct raid_set *rs)
+{
+	LOG_ERR(lc, 0, "spare set");
+}
+
+/* Push path and offset onto a table. */
+static int _dm_path_offset(struct lib_context *lc, char **table,
+			   int valid, const char *path, uint64_t offset)
+{
+	return p_fmt(lc, table, " %s %U",
+		     valid ? path : lc->path.error, offset);
+}
+
+/*
+ * Create dm table for linear mapping.
+ */
+static int _dm_linear(struct lib_context *lc, char **table, int valid,
+		      const char *path, uint64_t start, uint64_t sectors,
+		      uint64_t offset)
+{
+	return p_fmt(lc, table, "%U %U %s", start, sectors,
+		     get_dm_type(lc, t_linear)) ?
+		     _dm_path_offset(lc, table, valid, path, offset) : 0;
+}
+
+static int dm_linear(struct lib_context *lc, char **table,
+		     struct raid_set *rs)
+{
+	unsigned int segments = 0;
+	uint64_t start = 0, sectors = 0;
+	struct raid_dev *rd;
+	struct raid_set *r;
+
+	/* Stacked linear sets. */
+	list_for_each_entry(r, &rs->sets, list) {
+		if (!T_SPARE(r)) {
+			int ret;
+			char *path;
+
+			if (!(path = mkdm_path(lc, r->name)))
+				goto err;
+	
+			sectors = total_sectors(lc, r);
+			ret = _dm_linear(lc, table, valid_rs(r), path,
+					 start, sectors, 0);
+			dbg_free(path);
+			segments++;
+			start += sectors;
+	
+			if (!ret ||
+			    (r->sets.next != &rs->sets &&
+			     !p_fmt(lc, table, "\n")))
+				goto err;
+		}
+	}
+
+	/* Devices of a linear set. */
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (!T_SPARE(rd)) {
+			if (!_dm_linear(lc, table, valid_rd(rd), rd->di->path,
+					start, rd->sectors, rd->offset))
+				goto err;
+	
+			segments++;
+			start += rd->sectors;
+
+			if (rd->devs.next != &rs->devs &&
+			    !p_fmt(lc, table, "\n"))
+				goto err;
+		}
+	}
+
+	return segments ? 1 : 0;
+
+   err:
+	return log_alloc_err(lc, __func__);
+}
+
+/*
+ * Create dm table for a partition mapping.
+ *
+ * Partitioned RAID set with 1 RAID device
+ * defining a linear partition mapping.
+ */
+static int dm_partition(struct lib_context *lc, char **table,
+			struct raid_set *rs)
+{
+	return dm_linear(lc, table, rs);
+}
+
+/*
+ * Create dm table for striped mapping taking
+ * different disk sizes and the stride size into acccount.
+ *
+ * If metadata format handler requests a maximized mapping,
+ * more than one mapping table record will be created and
+ * stride boundaries will get paid attention to.
+ *
+ * Eg, 3 disks of 80, 100, 120 GB capacity:
+ *
+ * 0     240GB striped /dev/sda 0 /dev/sdb 0 /dev/sdc 0 
+ * 240GB 40GB  striped /dev/sdb 80GB /dev/sdc 80GB
+ * 280GB 20GB  linear /dev/sdc 100GB
+ *
+ */
+/* Push begin of line onto a RAID0 table. */
+static int _dm_raid0_bol(struct lib_context *lc, char **table,
+			 uint64_t min, uint64_t last_min,
+			 unsigned int n, unsigned int stride)
+{
+	return p_fmt(lc, table,
+		     n > 1 ? "%U %U %s %u %u" : "%U %U %s",
+		     last_min * n, (min - last_min) * n,
+		     get_dm_type(lc, n > 1 ? t_raid0 : t_linear),
+		     n, stride);
+}
+
+/* Push end of line onto a RAID0 table. */
+static int _dm_raid0_eol(struct lib_context *lc,
+			 char **table, struct raid_set *rs,
+			 unsigned int *stripes, uint64_t last_min)
+{
+	struct raid_set *r;
+	struct raid_dev *rd;
+
+	/* Stacked striped sets. */
+	list_for_each_entry(r, &rs->sets, list) {
+		if (total_sectors(lc, r) > last_min) {
+			int ret;
+			char *path;
+
+			if (!(path = mkdm_path(lc, r->name)))
+				goto err;
+
+			ret = _dm_path_offset(lc, table, valid_rs(r),
+					      path, last_min);
+			dbg_free(path);
+
+			if (!ret)
+				goto err;
+
+			(*stripes)++;
+		}
+	}
+	
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (!T_SPARE(rd) &&
+		    rd->sectors > last_min &&
+		    !_dm_path_offset(lc, table, valid_rd(rd), rd->di->path,
+				     rd->offset + last_min))
+			goto err;
+
+		(*stripes)++;
+	}
+
+	return 1;
+
+   err:
+	return 0;
+}
+
+/* Count RAID sets/devices larger than given minimum size. */
+static unsigned int _dm_raid_devs(struct lib_context *lc,
+				  struct raid_set *rs, uint64_t min)
+{
+	unsigned int ret = 0;
+	struct raid_set *r;
+	struct raid_dev *rd;
+
+	/* Stacked sets. */
+	list_for_each_entry(r, &rs->sets, list) {
+		if (!T_SPARE(r) && total_sectors(lc, r) > min)
+			ret++;
+	}
+	
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (!T_SPARE(rd) && rd->sectors > min)
+			ret++;
+	}
+
+	return ret;
+}
+	
+static int dm_raid0(struct lib_context *lc, char **table,
+		    struct raid_set *rs)
+{
+	unsigned int stripes = 0;
+	uint64_t min, last_min = 0;
+
+	for (; (min = _smallest(lc, rs, last_min)); last_min = min) {
+		if (last_min && !p_fmt(lc, table, "\n"))
+			goto err;
+
+		if (!_dm_raid0_bol(lc, table, round_down(min, rs->stride),
+				   last_min, _dm_raid_devs(lc, rs, last_min),
+				   rs->stride) ||
+		    !_dm_raid0_eol(lc, table, rs, &stripes, last_min))
+			goto err;
+
+		if (!F_MAXIMIZE(rs))
+			break;
+	}
+
+	return stripes ? 1 : 0;
+
+   err:
+	return log_alloc_err(lc, __func__);
+}
+
+/*
+ * Create dm table for mirrored mapping.
+ */
+
+/* Calculate dirty log region size. */
+static unsigned int calc_region_size(struct lib_context *lc, uint64_t sectors)
+{
+	const unsigned int mb_128 = 128*2*1024;
+	unsigned int max, region_size;
+
+	if ((max = sectors / 1024) > mb_128)
+		max = mb_128;
+
+	for (region_size = 128; region_size < max; region_size <<= 1);
+
+	return region_size >> 1;
+}
+
+static unsigned int get_rds(struct raid_set *rs, int valid)
+{
+	unsigned int ret = 0;
+	struct raid_dev *rd;
+
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (valid) {
+			if (valid_rd(rd))
+				ret++;
+		} else
+			ret++;
+	}
+
+	return ret;
+}
+
+static unsigned int get_dm_devs(struct raid_set *rs, int valid)
+{
+	unsigned int ret = 0;
+	struct raid_set *r;
+
+	/* Stacked mirror sets. */
+	list_for_each_entry(r, &rs->sets, list) {
+		if (valid) {
+			if (valid_rs(r))
+				ret++;
+		} else
+			ret++;
+	}
+
+	ret+= get_rds(rs, valid);
+
+	return ret;
+}
+
+/* Push begin of line onto a RAID1 table. */
+/* FIXME: persistent dirty log. */
+static int _dm_raid1_bol(struct lib_context *lc, char **table,
+			 struct raid_set *rs,
+			 uint64_t sectors, unsigned int mirrors)
+{
+	return (p_fmt(lc, table, "0 %U %s core 2 %u %s %u",
+		      sectors, get_dm_type(lc, t_raid1),
+		      calc_region_size(lc, sectors),
+		      (S_INCONSISTENT(rs->status) || S_NOSYNC(rs->status)) ?
+		      "sync" : "nosync", mirrors));
+}
+
+static int dm_raid1(struct lib_context *lc, char **table, struct raid_set *rs)
+{
+	uint64_t sectors = 0;
+	unsigned int mirrors = get_dm_devs(rs, 1);
+	struct raid_set *r;
+	struct raid_dev *rd;
+
+	switch (mirrors) {
+	case 0:
+		return 0;
+
+	case 1:
+		/*
+		 * In case we only have one mirror left,
+		 * a linear mapping will do.
+		 */
+		log_err(lc, "creating degraded mirror mapping for \"%s\"",
+			rs->name);
+		return dm_linear(lc, table, rs);
+	}
+		
+	if (!(sectors = _smallest(lc, rs, 0)))
+		LOG_ERR(lc, 0, "can't find smallest mirror!");
+
+	if (!_dm_raid1_bol(lc, table, rs, sectors, mirrors))
+		goto err;
+
+	/* Stacked mirror sets. */
+	list_for_each_entry(r, &rs->sets, list) {
+		if (valid_rs(r)) {
+			int ret;
+			char *path;
+
+			if (!(path = mkdm_path(lc, r->name)))
+				goto err;
+
+			ret = _dm_path_offset(lc, table, 1, path, 0);
+			dbg_free(path);
+
+			if (!ret)
+				goto err;
+		}
+	}
+
+	/* Lowest level mirror devices */
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (valid_rd(rd) &&
+		    !_dm_path_offset(lc, table, 1, rd->di->path, rd->offset))
+			goto err;
+	}
+
+	return 1;
+
+   err:
+	return log_alloc_err(lc, __func__);
+}
+
+/*
+ * Create dm table for RAID5 mapping.
+ */
+
+/* Push begin of line onto a RAID5 table. */
+/* FIXME: persistent dirty log. */
+static int _dm_raid45_bol(struct lib_context *lc, char **table,
+			 struct raid_set *rs,
+			 uint64_t sectors, unsigned int members)
+{
+	return p_fmt(lc, table, "0 %U %s core 2 %u %s %s 1 %u %u -1",
+		     sectors, get_dm_type(lc, rs->type),
+		     calc_region_size(lc, total_sectors(lc, rs) / _dm_raid_devs(lc, rs, 0)),
+		     (S_INCONSISTENT(rs->status) || S_NOSYNC(rs->status)) ?
+		     "sync" : "nosync",
+		     get_type(lc, rs->type), rs->stride, members);
+}
+
+static int dm_raid45(struct lib_context *lc, char **table, struct raid_set *rs)
+{
+	uint64_t sectors = 0;
+	unsigned int members = get_dm_devs(rs, 0);
+	struct raid_dev *rd;
+	struct raid_set *r;
+
+	if (!(sectors = _smallest(lc, rs, 0)))
+		LOG_ERR(lc, 0, "can't find smallest RAID4/5 member!");
+
+	/* Adjust sectors with chunk size: only whole chunks count */
+	sectors = sectors / rs->stride * rs->stride;
+
+	/*
+	 * Multiply size of smallest member by the number of data
+	 * devices to get the total sector count for the mapping.
+	 */
+	sectors *= members - 1;
+
+	if (!_dm_raid45_bol(lc, table, rs, sectors, members))
+		goto err;
+
+	/* Stacked RAID sets (for RAID50 etc.) */
+	list_for_each_entry(r, &rs->sets, list) {
+		int ret;
+		char *path;
+
+		if (!(path = mkdm_path(lc, r->name)))
+			goto err;
+
+		ret = _dm_path_offset(lc, table, valid_rs(r), path, 0);
+		dbg_free(path);
+
+		if (!ret)
+			goto err;
+	}
+
+	/* Lowest level RAID devices */
+	list_for_each_entry(rd, &rs->devs, devs) {
+		    if (!_dm_path_offset(lc, table, valid_rd(rd), rd->di->path,
+					 rd->offset))
+			goto err;
+	}
+
+	return 1;
+
+   err:
+	return log_alloc_err(lc, __func__);
+}
+
+/*
+ * Activate/deactivate (sub)sets.
+ */
+
+/*
+ * Array of handler functions for the various types.
+ */
+static struct type_handler {
+	const enum type type;
+	int(*f)(struct lib_context *lc, char **table, struct raid_set *rs);
+} type_handler[] = {
+	{ t_undef, dm_undef }, /* Needs to stay here! */
+	{ t_partition, dm_partition },
+	{ t_spare, dm_spare },
+	{ t_linear, dm_linear },
+	{ t_raid0, dm_raid0 },
+	{ t_raid1, dm_raid1 },
+	{ t_raid4, dm_raid45 },
+	{ t_raid5_ls, dm_raid45 },
+	{ t_raid5_rs, dm_raid45 },
+	{ t_raid5_la, dm_raid45 },
+	{ t_raid5_ra, dm_raid45 },
+	/* RAID types below not supported (yet) */
+	{ t_raid6, dm_unsup },
+};
+
+/* Retrieve type handler from array. */
+static struct type_handler *handler(struct raid_set *rs)
+{
+	struct type_handler *th = type_handler;
+
+	do {
+		if (rs->type == th->type)
+			return th;
+	} while (th++ < ARRAY_END(type_handler));
+
+	return type_handler;
+}
+
+/* Return mapping table */
+char *libdmraid_make_table(struct lib_context *lc, struct raid_set *rs)
+{
+	char *ret = NULL;
+
+	if (T_GROUP(rs))
+		return NULL;
+
+	if (!(handler(rs))->f(lc, &ret, rs))
+		LOG_ERR(lc, NULL, "no mapping possible for RAID set %s",
+			rs->name);
+
+	return ret;
+}
+
+
+enum dm_what { DM_ACTIVATE, DM_REGISTER};
+
+/* Register devices of the RAID set with the dmeventd. */
+/* REMOVEME: dummy functions once linking to the real ones. */
+#define	ALL_EVENTS 0xffffffff
+static int dm_register_for_event(char *a, char *b, int c)
+{
+	return 1;
+}
+
+static int dm_unregister_for_event(char *a, char *b, int c)
+{
+	return 1;
+}
+
+static int do_device(struct lib_context *lc, struct raid_set *rs,
+		     int (*f)()) // char *, char *, enum event_type))
+{
+	int ret = 0;
+	struct raid_dev *rd;
+
+	if (OPT_TEST(lc))
+		return 1;
+
+	return 1; /* REMOVEME: */
+
+	list_for_each_entry(rd, &rs->devs, devs) {
+		if (!(ret = f("dmraid", rd->di->path, ALL_EVENTS)))
+			break;
+	}
+
+	return ret ? 1 : 0;
+}
+
+static int register_devices(struct lib_context *lc, struct raid_set *rs)
+{
+	return do_device(lc, rs, dm_register_for_event);
+}
+
+/* Unregister devices of the RAID set with the dmeventd. */
+static int unregister_devices(struct lib_context *lc, struct raid_set *rs)
+{
+	return do_device(lc, rs, dm_unregister_for_event);
+}
+
+/* Activate a single set. */
+static int activate_subset(struct lib_context *lc, struct raid_set *rs,
+			   enum dm_what what)
+{
+	int ret = 0;
+	char *table = NULL;
+
+	if (T_GROUP(rs))
+		return 1;
+
+	if (what == DM_REGISTER)
+		return register_devices(lc, rs);
+
+	/* Call type handler */
+	if ((ret = (handler(rs))->f(lc, &table, rs))) {
+		if (OPT_TEST(lc))
+			display_table(lc, rs->name, table);
+		else
+			ret = dm_create(lc, rs, table);
+	} else
+		log_err(lc, "no mapping possible for RAID set %s", rs->name);
+
+	free_string(lc, &table);
+
+	return ret;
+}
+
+/* Activate a RAID set recursively (eg, RAID1 on top of RAID0). */
+static int activate_set(struct lib_context *lc, struct raid_set *rs,
+			enum dm_what what)
+{
+	struct raid_set *r;
+
+	if (!OPT_TEST(lc) &&
+	    what == DM_ACTIVATE &&
+	    dm_status(lc, rs)) {
+		log_print(lc, "RAID set \"%s\" already active", rs->name);
+		return 1;
+	}
+
+	/* Recursively walk down the chain of stacked RAID sets */
+	list_for_each_entry(r, &rs->sets, list) {
+		/* Activate set below this one */
+		if (!activate_set(lc, r, what) && !T_GROUP(rs))
+			return 0;
+	}
+
+	return activate_subset(lc, rs, what);
+}
+
+/* Deactivate a single set (one level of a device stack). */
+static int deactivate_superset(struct lib_context *lc, struct raid_set *rs,
+			       enum dm_what what)
+{
+	int ret = 1, status;
+
+	if (what == DM_REGISTER)
+		return unregister_devices(lc, rs);
+		
+	status = dm_status(lc, rs);
+	if (OPT_TEST(lc))
+		log_print(lc, "%s [%sactive]", rs->name, status ? "" : "in");
+	else if (status)
+		ret = dm_remove(lc, rs);
+	else {
+		log_print(lc, "RAID set \"%s\" is not active", rs->name);
+		ret = 1;
+	}
+
+	return ret;
+}
+
+/* Deactivate a RAID set. */
+static int deactivate_set(struct lib_context *lc, struct raid_set *rs,
+			  enum dm_what what)
+{
+	struct raid_set *r;
+
+	/*
+	 * Deactivate myself if not a group set,
+	 * which gets never activated itself.
+	 */
+	if (!T_GROUP(rs) &&
+	    !deactivate_superset(lc, rs, what))
+		return 0;
+
+	/* Deactivate any subsets recursively. */
+	list_for_each_entry(r, &rs->sets, list) {
+		if (!deactivate_set(lc, r, what))
+			return 0;
+	}
+
+	return 1;
+}
+
+
+/* External (de)activate interface. */
+int change_set(struct lib_context *lc, enum activate_type what, void *v)
+{
+	int ret = 0;
+	struct raid_set *rs = v;
+
+	switch (what) {
+	case A_ACTIVATE:
+		ret = activate_set(lc, rs, DM_ACTIVATE) &&
+		      activate_set(lc, rs, DM_REGISTER);
+		break;
+
+	case A_DEACTIVATE:
+		ret = deactivate_set(lc, rs, DM_REGISTER) &&
+		      deactivate_set(lc, rs, DM_ACTIVATE);
+	}
+
+	return ret;
+}
/cvs/dm/dmraid/lib/activate/activate.h,v  -->  standard output
revision 1.1
--- dmraid/lib/activate/activate.h
+++ -	2008-02-22 16:57:39.411945000 +0000
@@ -0,0 +1,18 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _ACTIVATE_H_
+#define _ACTIVATE_H_
+
+enum activate_type {
+	A_ACTIVATE,
+	A_DEACTIVATE,
+};
+
+int change_set(struct lib_context *lc, enum activate_type what, void *rs);
+
+#endif
/cvs/dm/dmraid/lib/activate/devmapper.c,v  -->  standard output
revision 1.1
--- dmraid/lib/activate/devmapper.c
+++ -	2008-02-22 16:57:39.495418000 +0000
@@ -0,0 +1,235 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * dmraid device-mapper lib interface functions.
+ */
+
+#include <libdevmapper.h>
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <ctype.h>
+#include <dirent.h>
+#include <errno.h>
+#include <unistd.h>
+
+#include "internal.h"
+#include "devmapper.h"
+
+/* Make up a dm path. */
+char *mkdm_path(struct lib_context *lc, const char *name)
+{
+	char *ret;
+	const char *dir = dm_dir();
+
+	if ((ret = dbg_malloc(strlen(dir) + strlen(name) + 2)))
+		sprintf(ret, "%s/%s", dir, name);
+	else
+		log_alloc_err(lc, __func__);
+
+	return ret;
+}
+
+/* Device-mapper NULL log function. */
+static void dmraid_log(int level, const char *file, int line,
+		       const char *f, ...)
+{
+	return;
+}
+
+/* Init device-mapper library. */
+static void _init_dm(void)
+{
+	dm_log_init(dmraid_log);
+}
+
+/* Cleanup at exit. */
+static void _exit_dm(struct dm_task *dmt)
+{
+	if (dmt)
+		dm_task_destroy(dmt);
+
+	dm_lib_release();
+	dm_lib_exit();
+}
+
+/*
+ * Retrieve list of registered mapping targets.
+ *
+ * dm-library must get inititalized by caller.
+ */
+static struct dm_versions *get_target_list(void)
+{
+	struct dm_task *dmt;
+
+	return (dmt = dm_task_create(DM_DEVICE_LIST_VERSIONS)) &&
+	       dm_task_run(dmt) ? dm_task_get_versions(dmt) : NULL;
+}
+
+/* Check a target's name against registered ones. */
+static int valid_ttype(struct lib_context *lc, char *ttype,
+		       struct dm_versions *targets)
+{
+	struct dm_versions *t, *last;
+
+	/*
+	 * If we don't have the list of target types registered
+	 * with the device-mapper core -> carry on and potentially
+	 * fail on target addition.
+	 */
+	if (!targets)
+		return 1;
+
+	/* Walk registered mapping target name list. */
+	t = targets;
+	do {
+		if (!strcmp(ttype, t->name))
+			return 1;
+
+		last = t;
+		t = (void*) t + t->next;
+	} while (last != t);
+
+	LOG_ERR(lc, 0, "device-mapper target type \"%s\" not in kernel", ttype);
+}
+
+/*
+ * Parse a mapping table and create the appropriate targets or
+ * check that a target type is registered with the device-mapper core.
+ */
+static int handle_table(struct lib_context *lc, struct dm_task *dmt,
+			char *table, struct dm_versions *targets)
+{
+	int line = 0, n, ret = 0;
+	uint64_t start, size;
+	char *nl = table, *p, ttype[32];
+
+	do {
+		p = nl;
+		line++;
+
+		/*
+		 * Not using sscanf string allocation
+		 * because it's not available in dietlibc.
+		 */
+		*ttype = 0;
+		if (sscanf(p, "%" PRIu64 " %" PRIu64 " %31s %n",
+			   &start, &size, ttype, &n) < 3)
+			LOG_ERR(lc, 0, "Invalid format in table line %d", line);
+
+		if (!(ret = valid_ttype(lc, ttype, targets)))
+			break;
+
+		nl = remove_delimiter((p += n), '\n');
+		if (dmt)
+			ret = dm_task_add_target(dmt, start, size, ttype, p);
+
+		add_delimiter(&nl, '\n');
+	} while (nl && ret);
+
+	return ret;
+}
+
+/* Parse a mapping table and create the appropriate targets. */
+static int parse_table(struct lib_context *lc, struct dm_task *dmt, char *table)
+{
+	return handle_table(lc, dmt, table, NULL);
+}
+
+/* Check if a target type is not registered with the kernel after a failure. */
+static int check_table(struct lib_context *lc, char *table)
+{
+	return handle_table(lc, NULL, table, get_target_list());
+}
+
+/* Create a mapped device. */
+int dm_create(struct lib_context *lc, struct raid_set *rs, char *table)
+{
+	int ret = 0;
+	struct dm_task *dmt;
+
+	_init_dm();
+
+	/* Create <dev_name> */
+	ret = (dmt = dm_task_create(DM_DEVICE_CREATE)) &&
+	       dm_task_set_name(dmt, rs->name) &&
+	       parse_table(lc, dmt, table) &&
+	       dm_task_run(dmt);
+
+	/*
+	 * In case device creation failed, check if target
+	 * isn't registered with the device-mapper core
+	 */
+	if (!ret)
+		check_table(lc, table);
+
+	_exit_dm(dmt);
+
+	return ret;
+}
+
+/* Remove a mapped device. */
+int dm_remove(struct lib_context *lc, struct raid_set *rs)
+{
+	int ret;
+	struct dm_task *dmt;
+
+	_init_dm();
+
+	/* remove <dev_name> */
+	ret = (dmt = dm_task_create(DM_DEVICE_REMOVE)) &&
+	       dm_task_set_name(dmt, rs->name) &&
+	       dm_task_run(dmt);
+
+	_exit_dm(dmt);
+
+	return ret;
+}
+
+/* Retrieve status of a mapped device. */
+/* FIXME: more status for device monitoring... */
+int dm_status(struct lib_context *lc, struct raid_set *rs)
+{
+	int ret;
+	struct dm_task *dmt;
+	struct dm_info info;
+
+	_init_dm();
+
+	/* Status <dev_name>. */
+	ret = (dmt = dm_task_create(DM_DEVICE_STATUS)) &&
+	       dm_task_set_name(dmt, rs->name) &&
+	       dm_task_run(dmt) &&
+	       dm_task_get_info(dmt, &info) &&
+	       info.exists;
+
+	_exit_dm(dmt);
+
+	return ret;
+}
+
+/* Retrieve device-mapper driver version. */
+int dm_version(struct lib_context *lc, char *version, size_t size)
+{
+	int ret;
+	struct dm_task *dmt;
+
+	/* Be prepared for device-mapper not in kernel. */
+	strncpy(version, "unknown", size);
+
+	_init_dm();
+
+	ret = (dmt = dm_task_create(DM_DEVICE_VERSION)) &&
+	       dm_task_run(dmt) &&
+	       dm_task_get_driver_version(dmt, version, size);
+
+	_exit_dm(dmt);
+
+	return ret;
+}
/cvs/dm/dmraid/lib/activate/devmapper.h,v  -->  standard output
revision 1.1
--- dmraid/lib/activate/devmapper.h
+++ -	2008-02-22 16:57:39.574643000 +0000
@@ -0,0 +1,17 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _DEVMAPPER_H_
+#define _DEVMAPPER_H
+
+char *mkdm_path(struct lib_context *lc, const char *name);
+int dm_create(struct lib_context *lc, struct raid_set *rs, char *table);
+int dm_remove(struct lib_context *lc, struct raid_set *rs);
+int dm_status(struct lib_context *lc, struct raid_set *rs);
+int dm_version(struct lib_context *lc, char *version, size_t size);
+
+#endif
/cvs/dm/dmraid/lib/datastruct/byteorder.h,v  -->  standard output
revision 1.1
--- dmraid/lib/datastruct/byteorder.h
+++ -	2008-02-22 16:57:39.656066000 +0000
@@ -0,0 +1,59 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/* Cheers utils.h */
+
+#ifndef _BYTEORDER_H
+#define _BYTEORDER_H
+
+#ifdef __KLIBC__
+#include <endian.h>
+#endif
+
+#ifdef	DM_BYTEORDER_SWAB
+
+static inline uint64_t le64_to_cpu(uint64_t x)
+{
+	return((((uint64_t)x & 0x00000000000000ffULL) << 56) | 
+	       (((uint64_t)x & 0x000000000000ff00ULL) << 40) | 
+	       (((uint64_t)x & 0x0000000000ff0000ULL) << 24) | 
+	       (((uint64_t)x & 0x00000000ff000000ULL) <<  8) |
+	       (((uint64_t)x & 0x000000ff00000000ULL) >>  8) | 
+	       (((uint64_t)x & 0x0000ff0000000000ULL) >> 24) | 
+	       (((uint64_t)x & 0x00ff000000000000ULL) >> 40) | 
+	       (((uint64_t)x & 0xff00000000000000ULL) >> 56));
+}
+
+static inline int32_t le32_to_cpu(int32_t x)
+{
+	return((((u_int32_t)x & 0x000000ffU) << 24) | 
+	       (((u_int32_t)x & 0x0000ff00U) <<  8) | 
+	       (((u_int32_t)x & 0x00ff0000U) >>  8) | 
+	       (((u_int32_t)x & 0xff000000U) >> 24));
+}
+
+static inline int16_t le16_to_cpu(int16_t x)
+{
+	return((((u_int16_t)x & 0x00ff) << 8) | 
+	       (((u_int16_t)x & 0xff00) >> 8));
+}
+
+#define	CVT64(x)	do { x = le64_to_cpu(x); } while(0)
+#define	CVT32(x)	do { x = le32_to_cpu(x); } while(0)
+#define	CVT16(x)	do { x = le16_to_cpu(x); } while(0)
+
+#else
+
+#define CVT64(x)
+#define	CVT32(x)
+#define	CVT16(x)
+
+#undef	DM_BYTEORDER_SWAB
+
+#endif /* #ifdef DM_BYTEORDER_SWAB */
+
+#endif
/cvs/dm/dmraid/lib/device/ata.c,v  -->  standard output
revision 1.1
--- dmraid/lib/device/ata.c
+++ -	2008-02-22 16:57:39.734699000 +0000
@@ -0,0 +1,38 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/* Thx scsiinfo. */
+
+#include <string.h>
+#include <sys/ioctl.h>
+#include <sys/types.h>
+#include <mm/dbg_malloc.h>
+
+#include "dev-io.h"
+#include "ata.h"
+
+int get_ata_serial(struct lib_context *lc, int fd, struct dev_info *di)
+{
+	int ret = 0;
+	const int cmd_offset = 4;
+	unsigned char *buf;
+	struct ata_identify *ata_ident;
+
+	if ((buf = dbg_malloc(cmd_offset + sizeof(*ata_ident)))) {
+		buf[0] = ATA_IDENTIFY_DEVICE;
+		buf[3] = 1;
+		if (!ioctl(fd, HDIO_DRIVE_CMD, buf)) {
+			ata_ident = (struct ata_identify*) &buf[cmd_offset];
+			if ((di->serial = dbg_strdup(remove_white_space(lc, (char*) ata_ident->serial, ATA_SERIAL_LEN))))
+				ret = 1;
+		}
+	
+		dbg_free(buf);
+	}
+
+	return ret;
+}
/cvs/dm/dmraid/lib/device/ata.h,v  -->  standard output
revision 1.1
--- dmraid/lib/device/ata.h
+++ -	2008-02-22 16:57:39.814157000 +0000
@@ -0,0 +1,42 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/* Thx scsiinfo */
+
+#ifndef	_ATA_H_
+#define	_ATA_H_
+
+struct ata_identify {
+	unsigned short dummy[10];
+#define	ATA_SERIAL_LEN	20
+	unsigned char  serial[ATA_SERIAL_LEN];
+	unsigned short dummy1[3];
+	unsigned char  fw_rev[8];
+	unsigned char  model[40];
+	unsigned short dummy2[33];
+	unsigned short major_rev_num;
+	unsigned short minor_rev_num;
+	unsigned short command_set_1;
+	unsigned short command_set_2;
+	unsigned short command_set_extension;
+	unsigned short cfs_enable_1;
+	unsigned short dummy3;
+	unsigned short csf_default;
+	unsigned short dummy4[168];
+};
+
+#ifndef ATA_IDENTIFY_DEVICE
+#define ATA_IDENTIFY_DEVICE 0xEC
+#endif
+#ifndef HDIO_DRIVE_CMD
+#define HDIO_DRIVE_CMD    0x031F
+#endif
+
+struct lib_context;
+int get_ata_serial(struct lib_context *lc, int fd, struct dev_info *di);
+
+#endif
/cvs/dm/dmraid/lib/device/dev-io.h,v  -->  standard output
revision 1.1
--- dmraid/lib/device/dev-io.h
+++ -	2008-02-22 16:57:39.894030000 +0000
@@ -0,0 +1,23 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_DEV_IO_H_
+#define	_DEV_IO_H_
+
+#include <linux/hdreg.h>
+#include <sys/stat.h>
+#include "internal.h"
+
+#define BLKGETSIZE	_IO(0x12, 0x60) /* get block device size */
+#define BLKSSZGET	_IO(0x12, 0x68) /* get block device sector size */
+
+#define	DMRAID_SECTOR_SIZE	512
+
+int discover_devices(struct lib_context *lc, char **devnodes);
+int removable_device(struct lib_context *lc, char *dev_path);
+
+#endif
/cvs/dm/dmraid/lib/device/scan.c,v  -->  standard output
revision 1.1
--- dmraid/lib/device/scan.c
+++ -	2008-02-22 16:57:39.977011000 +0000
@@ -0,0 +1,336 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifdef __KLIBC__
+# define __KERNEL_STRICT_NAMES
+# include <dirent.h>
+# include <paths.h>
+#else
+# include <dirent.h>
+# include <mntent.h>
+#endif
+
+#include <stdlib.h>
+#include <linux/hdreg.h>
+#include <sys/ioctl.h>
+#include "internal.h"
+#include "ata.h"
+#include "scsi.h"
+
+/*
+ * subdirectory below sysfs moint point holding the 
+ * subdirectory hierarchies of all block devices.
+ */
+#define	BLOCK 		"/block"
+
+/* Find sysfs mount point */
+#ifndef	_PATH_MOUNTS
+#define	_PATH_MOUNTS	"/proc/mounts"
+#endif
+
+static char *find_sysfs_mp(struct lib_context *lc)
+{
+#ifndef __KLIBC__
+	char *ret = NULL;
+	FILE *mfile;
+	struct mntent *ment;
+
+	/* Try /proc/mounts first and failback to /etc/mtab. */
+	if (!(mfile = setmntent(_PATH_MOUNTS, "r"))) {
+		if (!(mfile = setmntent(_PATH_MOUNTED, "r")))
+			LOG_ERR(lc, NULL, "Unable to open %s or %s",
+				_PATH_MOUNTS, _PATH_MOUNTED);
+	}
+
+	while ((ment = getmntent(mfile))) {
+		if (!strcmp(ment->mnt_type, "sysfs")) {
+			ret = ment->mnt_dir;
+			break;
+		}
+	};
+
+	endmntent(mfile);
+
+	return ret;
+#else
+	return (char*) "/sys";
+#endif
+}
+
+/* Make up an absolute sysfs path given a relative one. */
+static char *mk_sysfs_path(struct lib_context *lc, char const *path) {
+	static char *ret = NULL, *sysfs_mp;
+
+	if (!(sysfs_mp = find_sysfs_mp(lc)))
+		LOG_ERR(lc, NULL, "finding sysfs mount point");
+
+	if ((ret = dbg_malloc(strlen(sysfs_mp) + strlen(path) + 1)))
+		sprintf(ret, "%s%s", sysfs_mp, path);
+	else
+		log_alloc_err(lc, __func__);
+
+	return ret;
+}
+
+/* Test with sparse mapped devices. */
+#ifdef	DMRAID_TEST
+static int dm_test_device(struct lib_context *lc, char *path)
+{
+	struct stat s;
+
+	return !lstat(path, &s) &&
+	       S_ISLNK(s.st_mode) &&
+	       !strncmp(get_basename(lc, path), "dm-", 3);
+}
+
+/* Fake a SCSI serial number by reading it from a file. */
+static int get_dm_test_serial(struct lib_context *lc,
+			      struct dev_info *di, char *path)
+{
+	int ret = 1;
+	char *serial, buffer[32];
+	const char *dot_serial = ".serial";
+	FILE *f;
+
+	if (!(serial = dbg_malloc(strlen(path) + strlen(dot_serial) + 1)))
+		return log_alloc_err(lc, __func__);
+
+	sprintf(serial, "%s%s", path, dot_serial);
+	if ((f = fopen(serial, "r")) &&
+	    fgets(buffer, 31, f) &&
+	    !(di->serial = dbg_strdup(remove_white_space(lc, buffer,
+							 strlen(buffer)))))
+		ret = 0;
+
+	dbg_free(serial);
+	if (f)
+		fclose(f);
+	else
+		log_warn(lc, "missing dm serial file for %s", di->path);
+
+	return ret;
+#undef	DOT_SERIAL
+}
+#endif
+
+/*
+ * Ioctl for sector, optionally for device size
+ * and get device serial number.
+ */
+static int get_device_serial(struct lib_context *lc, int fd,
+			     struct dev_info *di)
+{
+	/*
+	 * In case new generic SCSI ioctl fails,
+	 * try ATA and fall back to old SCSI ioctl.
+	 */
+	return get_scsi_serial(lc, fd, di, SG) || /* SG: generic scsi ioctl. */
+	       get_ata_serial(lc, fd, di) ||	  /* Get ATA serial number. */
+	       get_scsi_serial(lc, fd, di, OLD);  /* OLD: Old scsi ioctl. */
+}
+
+static int di_ioctl(struct lib_context *lc, int fd, struct dev_info *di)
+{
+	unsigned int sector_size = 0;
+	unsigned long size;
+
+	/* Fetch sector size. */
+	if (ioctl(fd, BLKSSZGET, &sector_size))
+		sector_size = DMRAID_SECTOR_SIZE;
+
+	if (sector_size != DMRAID_SECTOR_SIZE)
+		LOG_ERR(lc, 0, "unsupported sector size %d on %s.",
+			sector_size, di->path);
+
+	/* Use size device ioctl in case we didn't get the size from sysfs. */
+	if (!di->sectors && !ioctl(fd, BLKGETSIZE, &size))
+		di->sectors = size;
+
+#ifdef	DMRAID_TEST
+	/* Test with sparse mapped devices. */
+	if (dm_test_device(lc, di->path))
+		return get_dm_test_serial(lc, di, di->path);
+	else
+#endif
+	return get_device_serial(lc, fd, di);
+}
+
+/* Are we interested in this device ? */
+static int interested(struct lib_context *lc, char *path)
+{
+	char *name = get_basename(lc, path);
+
+	/*
+	 * Whole IDE and SCSI disks only.
+	 */
+	return (!isdigit(name[strlen(name) - 1]) &&
+	       (*(name + 1) == 'd' && (*name == 'h' || *name == 's')))
+
+#ifdef	DMRAID_TEST
+	/*
+	 * Include dm devices for testing.
+	 */
+		|| dm_test_device(lc, path)
+#endif
+	;
+}
+
+/* Ask sysfs, if a device is removable. */
+int removable_device(struct lib_context *lc, char *dev_path)
+{
+	int ret = 0;
+	char buf[2], *name, *sysfs_path, *sysfs_file;
+	const char *sysfs_removable = "removable";
+	FILE *f;
+
+	if (!(sysfs_path = mk_sysfs_path(lc, BLOCK)))
+		return 0;
+
+	name = get_basename(lc, dev_path);
+	if (!(sysfs_file = dbg_malloc(strlen(sysfs_path) + strlen(name) +
+				      strlen(sysfs_removable) + 3))) {
+		log_alloc_err(lc, __func__);
+		goto out;
+	}
+
+	sprintf(sysfs_file, "%s/%s/%s", sysfs_path, name, sysfs_removable);
+	if ((f = fopen(sysfs_file, "r"))) {
+		/* Using fread for klibc compatibility. */
+		if (fread(buf, sizeof(char), sizeof(buf) - 1, f) &&
+		    *buf == '1') {
+			log_notice(lc, "skipping removable device %s",
+				   dev_path);
+			ret = 1;
+		}
+
+		fclose(f);
+	}
+
+	dbg_free(sysfs_file);
+
+   out:
+	dbg_free(sysfs_path);
+
+	return ret;
+}
+
+/*
+ * Read the size in sectors from the sysfs "size" file.
+ * Avoid access to removable devices.
+ */
+static int sysfs_get_size(struct lib_context *lc, struct dev_info *di,
+			  const char *path, char *name)
+{
+	int ret = 0;
+	char buf[22], *sysfs_file;
+	const char *sysfs_size = "size";
+	FILE *f;
+
+	if (!(sysfs_file = dbg_malloc(strlen(path) + strlen(name) +
+				      strlen(sysfs_size) + 3)))
+		return log_alloc_err(lc, __func__);
+
+	sprintf(sysfs_file, "%s/%s/%s", path, name, sysfs_size);
+	if ((f = fopen(sysfs_file, "r"))) {
+		/* Use fread+sscanf for klibc compatibility. */
+		if (fread(buf, sizeof(char), sizeof buf - 1, f) &&
+		    (ret = sscanf(buf, "%" PRIu64, &di->sectors)) != 1) {
+			ret = 0;
+			log_err(lc, "reading disk size for %s from sysfs",
+				di->path);
+		}
+
+		fclose(f);
+	} else
+		log_err(lc, "opening %s", sysfs_file);
+
+	dbg_free(sysfs_file);
+
+	return ret;
+}
+
+static int get_size(struct lib_context *lc, char *path,
+		    char *name, int sysfs)
+{
+	int fd, ret = 0;
+	char *dev_path;
+	struct dev_info *di = NULL;
+
+	if (!(dev_path = dbg_malloc(strlen(_PATH_DEV) + strlen(name) + 1)))
+		return log_alloc_err(lc, __func__);
+
+	sprintf(dev_path, "%s%s", _PATH_DEV, name);
+	if (!interested(lc, dev_path)) {
+		ret = 0;
+		goto out;
+	}
+
+	if (removable_device(lc, dev_path) ||
+	    !(di = alloc_dev_info(lc, dev_path)) ||
+	    (sysfs && !sysfs_get_size(lc, di, path, name)) ||
+	    (fd = open(dev_path, O_RDONLY)) == -1)
+		goto out;
+
+	if (di_ioctl(lc, fd, di)) {
+		list_add(&di->list, LC_DI(lc));
+		ret = 1;
+	}
+
+	close(fd);
+
+   out:
+	dbg_free(dev_path);
+
+	if (!ret && di)
+		free_dev_info(lc, di);
+
+	return ret;
+}
+
+/*
+ * Find disk devices in sysfs or directly
+ * in /dev (for Linux 2.4) and keep information.
+ */
+int discover_devices(struct lib_context *lc, char **devnodes)
+{
+	int sysfs, ret = 0;
+	char *path, *p;
+	DIR *d;
+	struct dirent *de;
+
+	if ((p = mk_sysfs_path(lc, BLOCK))) {
+		sysfs = 1;
+		path = p;
+	} else {
+		sysfs = 0;
+		path = (char*) _PATH_DEV;
+		log_print(lc, "carrying on with %s", path);
+	}
+
+	if (!(d = opendir(path))) {
+		log_err(lc, "opening path %s", path);
+		goto out;
+	}
+
+	if (devnodes && *devnodes) {
+		while (*devnodes)
+			get_size(lc, path, get_basename(lc, *devnodes++),
+				 sysfs);
+	} else {
+		while ((de = readdir(d)))
+			get_size(lc, path, de->d_name, sysfs);
+	}
+
+	closedir(d);
+	ret = 1;
+
+  out:
+	if (p)
+		dbg_free(p);
+
+	return ret;
+}
/cvs/dm/dmraid/lib/device/scsi.c,v  -->  standard output
revision 1.1
--- dmraid/lib/device/scsi.c
+++ -	2008-02-22 16:57:40.065183000 +0000
@@ -0,0 +1,113 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <string.h>
+#include <sys/ioctl.h>
+#include <sys/types.h>
+#include <mm/dbg_malloc.h>
+#include <scsi/scsi_ioctl.h>
+
+/* FIXME: sg header mess. */
+#include <scsi/sg.h>
+#include <scsi/scsi.h>
+
+#include "dev-io.h"
+#include "scsi.h"
+
+/* Thx scsiinfo. */
+
+/* Initialize SCSI inquiry command block (used both with SG and old ioctls). */
+static void set_cmd(unsigned char *cmd, size_t len)
+{
+	cmd[0] = 0x12;	/* INQUIRY */
+	cmd[1] = 1;
+	cmd[2] = 0x80;	/* page code: SCSI serial */
+	cmd[3] = 0;
+	cmd[4] = (unsigned char) (len & 0xff);
+	cmd[5] = 0;
+}
+
+/*
+ * SCSI SG_IO ioctl to get serial number of a unit.
+ */
+static int sg_inquiry(int fd, unsigned char *response, size_t response_len)
+{
+	unsigned char cmd[6];
+	struct sg_io_hdr io_hdr;
+
+	set_cmd(cmd, response_len);
+
+	/* Initialize generic (SG) SCSI ioctl header. */
+	memset(&io_hdr, 0, sizeof(io_hdr));
+	io_hdr.interface_id	= 'S';
+	io_hdr.cmdp		= cmd;
+	io_hdr.cmd_len		= sizeof(cmd);
+	io_hdr.sbp		= NULL;
+	io_hdr.mx_sb_len	= 0;
+	io_hdr.dxfer_direction	= SG_DXFER_FROM_DEV;
+	io_hdr.dxferp		= response;
+	io_hdr.dxfer_len	= response_len;
+	io_hdr.timeout		= 6000; /* [ms] */
+
+	return ioctl(fd, SG_IO, &io_hdr) ? 0 : 1;
+}
+
+/*
+ * Old SCSI ioctl as fallback to get serial number of a unit.
+ */
+static int old_inquiry(int fd, unsigned char *response, size_t response_len)
+{
+	unsigned int *i = (unsigned int*) response;
+
+	i[0] = 0; /* input data length */
+	i[1] = response_len; /* output buffer length */
+	set_cmd((unsigned char*) &i[2], response_len); 
+
+	return ioctl(fd, SCSI_IOCTL_SEND_COMMAND, response) ? 0 : 1;
+}
+
+/*
+ * Retrieve SCSI serial number.
+ */
+#define	MAX_RESPONSE_LEN	255
+int get_scsi_serial(struct lib_context *lc, int fd, struct dev_info *di,
+		    enum ioctl_type type)
+{
+	int ret = 0;
+	size_t actual_len;
+	unsigned char *response;
+	/*
+	 * Define ioctl function and offset into response buffer of serial
+	 * string length field (serial string follows length field immediately)
+	 */
+	struct {
+		int (*ioctl_func)(int, unsigned char*, size_t);
+		unsigned int start;
+	} param[] = {
+		{ sg_inquiry ,  3 },
+		{ old_inquiry, 11 },
+	}, *p = (SG == type) ? param : param + 1;
+
+	if (!(response = dbg_malloc(MAX_RESPONSE_LEN)))
+		return 0;
+
+	actual_len = p->start + 1;
+	if ((ret = (p->ioctl_func(fd, response, actual_len)))) {
+		size_t serial_len = (size_t) response[p->start];
+
+		if (serial_len > actual_len) {
+			actual_len += serial_len;
+			ret = p->ioctl_func(fd, response, actual_len);
+		}
+
+		ret = ret && (di->serial = dbg_strdup(remove_white_space(lc, (char*) &response[p->start + 1], serial_len)));
+	}
+
+	dbg_free(response);
+
+	return ret;
+}
/cvs/dm/dmraid/lib/device/scsi.h,v  -->  standard output
revision 1.1
--- dmraid/lib/device/scsi.h
+++ -	2008-02-22 16:57:40.146618000 +0000
@@ -0,0 +1,20 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_SCSI_H_
+#define	_SCSI_H_
+
+/* Ioctl types possible (SG = SCSI generic, OLD = old SCSI command ioctl. */
+enum ioctl_type {
+	SG,
+	OLD,
+};
+
+int get_scsi_serial(struct lib_context *lc, int fd,
+		    struct dev_info *di, enum ioctl_type type);
+
+#endif
/cvs/dm/dmraid/lib/display/display.c,v  -->  standard output
revision 1.1
--- dmraid/lib/display/display.c
+++ -	2008-02-22 16:57:40.244819000 +0000
@@ -0,0 +1,371 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include "internal.h"
+#include "activate/devmapper.h"
+
+#define	ARRAY_LIMIT(array, idx) \
+	((idx) < ARRAY_SIZE(array) ? (idx) : ARRAY_SIZE(array) - 1)
+
+struct log_handler {
+	const char *field;
+	const unsigned char minlen;
+	void (*log_func)(struct lib_context*, void *arg);
+	void *arg;
+};
+
+static void log_string(struct lib_context *lc, void *arg)
+{
+	log_print_nnl(lc, "%s", (char*) arg);
+}
+
+static void log_uint64(struct lib_context *lc, void *arg)
+{
+	log_print_nnl(lc, "%" PRIu64, *((uint64_t*) arg));
+}
+
+static void log_uint(struct lib_context *lc, void *arg)
+{
+	log_print_nnl(lc, "%u", *((unsigned int*) arg));
+}
+
+/* Log a structure member by field name. */
+static int log_field(struct lib_context *lc, const struct log_handler *lh,
+		     size_t lh_size, char *field)
+{
+	const struct log_handler *h;
+
+	for (h = lh; h < lh + lh_size; h++) {
+		size_t len = strlen(field);
+
+		if (!strncmp(field, h->field,
+			     len > h->minlen ? len : h->minlen)) {
+			h->log_func(lc, h->arg);
+			return 1;
+		}
+	}
+
+	log_print_nnl(lc, "*ERR*");
+
+	return 1;
+}
+
+/* Log a list of structure members by field name. */
+static void log_fields(struct lib_context *lc, const struct log_handler *lh,
+		       size_t lh_size) {
+	int logged = 0, last_logged = 0;
+	const char delim = *OPT_STR_SEPARATOR(lc);
+	char *p, *sep, *sep_sav;
+
+	if (!(sep_sav = dbg_strdup((char*) OPT_STR_COLUMN(lc)))) {
+		log_alloc_err(lc, __func__);
+		return;
+	}
+
+	sep = sep_sav;
+	do {
+		sep = remove_delimiter((p = sep), delim);
+		if (last_logged)
+			log_print_nnl(lc, "%c", delim);
+
+		last_logged = log_field(lc, lh, lh_size, p);
+		logged += last_logged;
+		add_delimiter(&sep, delim);
+	} while (sep);
+
+	dbg_free(sep_sav);
+	if (logged)
+		log_print(lc, "");
+}
+
+/* Display information about a block device */
+static void log_disk(struct lib_context *lc, struct list_head *pos)
+{
+	struct dev_info *di = list_entry(pos, typeof(*di), list);
+
+	if (OPT_STR_COLUMN(lc)) {
+		const struct log_handler log_handlers[] = {
+			{ "devpath", 1, log_string, di->path},
+			{ "path", 1, log_string, di->path},
+			{ "sectors", 3, log_uint64, &di->sectors},
+			{ "serialnumber", 3, log_string,
+			  di->serial ? (void*) di->serial : (void*) "N/A"},
+			{ "size", 2, log_uint64, &di->sectors},
+		};
+
+		log_fields(lc, log_handlers, ARRAY_SIZE(log_handlers));
+	} else {
+		const char *fmt[] = {
+			"%s: %12" PRIu64 " total, \"%s\"",
+			"%s",
+			"%s:%" PRIu64 ":\"%s\"",
+		};
+	
+		log_print(lc, fmt[ARRAY_LIMIT(fmt, OPT_COLUMN(lc))],
+			  di->path, di->sectors,
+			  di->serial ? di->serial : "N/A");
+	}
+}
+
+/* Turn NULL (= "unknown") into a displayable string. */
+static const char *check_null(const char *str)
+{
+	return str ? str : "unknown";
+}
+
+/* Log native RAID device information. */
+static void log_rd_native(struct lib_context *lc, struct list_head *pos)
+{
+	struct raid_dev *rd = list_entry(pos, typeof(*rd), list);
+
+	if (rd->fmt->log) {
+		rd->fmt->log(lc, rd);
+		log_print(lc, "");
+	} else
+		log_print(lc, "\"%s\" doesn't support native logging of RAID "
+			  "device information", rd->fmt->name);
+}
+
+/* Display information about a RAID device */
+static void log_rd(struct lib_context *lc, struct list_head *pos)
+{
+	struct raid_dev *rd = list_entry(pos, typeof(*rd), list);
+
+	if (OPT_STR_COLUMN(lc)) {
+		const struct log_handler log_handlers[] = {
+			{ "dataoffset", 2, log_uint64, &rd->offset},
+			{ "devpath", 2, log_string, rd->di->path },
+			{ "format", 1, log_string, (void*) rd->fmt->name },
+			{ "offset", 1, log_uint64, &rd->offset},
+			{ "path", 1, log_string, rd->di->path },
+			{ "raidname", 1, log_string, rd->name },
+			{ "type", 1, log_string,
+			  (void*) check_null(get_type(lc, rd->type)) },
+			{ "sectors", 2, log_uint64, &rd->sectors},
+			{ "size", 2, log_uint64, &rd->sectors},
+			{ "status", 2, log_string,
+			  (void*) check_null(get_status(lc, rd->status)) },
+		};
+
+		log_fields(lc, log_handlers, ARRAY_SIZE(log_handlers));
+	} else {
+		const char *fmt[] = {
+			"%s: %s, \"%s\", %s, %s, %" PRIu64 " sectors, "
+			"data@ %" PRIu64,
+			"%s",
+			"%s:%s:%s:%s:%s:%" PRIu64 ":%" PRIu64,
+		};
+	
+		log_print(lc, fmt[ARRAY_LIMIT(fmt, OPT_COLUMN(lc))],
+			  rd->di->path, rd->fmt->name, rd->name,
+			  check_null(get_type(lc, rd->type)),
+			  check_null(get_status(lc, rd->status)),
+			  rd->sectors, rd->offset);
+	}
+}
+
+/* Dispatch log functions. */
+static void log_devices(struct lib_context *lc, enum dev_type type)
+{
+	struct list_head *pos;
+	struct {
+		enum dev_type type;
+		struct list_head *list;
+		void (*log)(struct lib_context *, struct list_head *);
+	} types[] = {
+		{ DEVICE, LC_DI(lc), log_disk },
+		{ NATIVE, LC_RD(lc), log_rd_native },
+		{ RAID,   LC_RD(lc), log_rd },
+	}, *t = types;
+
+	do {
+		if (t->type == type) {
+			list_for_each(pos, t->list)
+				t->log(lc, pos);
+
+			return;
+		}
+	} while (t++ < ARRAY_END(types));
+
+	LOG_ERR(lc, , "%s: unknown device type", __func__);
+}
+
+/* Display information about a dmraid format handler */
+static void log_format(struct lib_context *lc, struct dmraid_format *fmt)
+{
+	log_print_nnl(lc, "%-7s : %s", fmt->name, fmt->descr);
+	if (fmt->caps)
+		log_print_nnl(lc, " (%s)", fmt->caps);
+
+	log_print(lc, "");
+}
+
+/* Pretty print a mapping table. */
+void display_table(struct lib_context *lc, char *rs_name, char *table)
+{
+	char *nl = table, *p;
+
+	do {
+		nl = remove_delimiter((p = nl), '\n');
+		log_print(lc, "%s: %s", rs_name, p);
+		add_delimiter(&nl, '\n');
+	} while (nl);
+}
+
+/* Display information about devices depending on device type. */
+void display_devices(struct lib_context *lc, enum dev_type type)
+{
+	int devs;
+
+	if ((devs = count_devices(lc, type))) {
+		log_info(lc, "%s device%s discovered:\n",
+			 (type & (RAID|NATIVE)) ? "RAID" : "Block",
+			  devs == 1 ? "" : "s");
+
+		log_devices(lc, type);
+	}
+}
+
+/* Retrieve format name from (hierarchical) raid set. */
+static void *get_format_name(struct raid_set *rs)
+{
+	struct dmraid_format *fmt = get_format(rs);
+
+	return (void*) check_null(fmt ? fmt->name : NULL);
+}
+
+static void log_rs(struct lib_context *lc, struct raid_set *rs)
+{
+	unsigned int devs = 0, spares = 0, subsets = 0;
+	uint64_t sectors = 0;
+
+	if (T_GROUP(rs) && !OPT_GROUP(lc))
+		return;
+	
+	sectors = total_sectors(lc, rs);
+	subsets = count_sets(lc, &rs->sets);
+	devs    = count_devs(lc, rs, ct_dev);
+	spares  = count_devs(lc, rs, ct_spare);
+	
+	if (OPT_STR_COLUMN(lc)) {
+		const struct log_handler log_handlers[] = {
+			{ "devices", 1, log_uint, &devs },
+			{ "format", 1, log_string, get_format_name(rs) },
+			{ "raidname", 1, log_string, rs->name },
+			{ "sectors", 2, log_uint64, &sectors },
+			{ "size", 2, log_uint64, &sectors },
+			{ "spares", 2, log_uint, &spares },
+			{ "status", 3, log_string,
+			   (void*) check_null(get_status(lc, rs->status)) },
+			{ "stride", 3, log_uint, &rs->stride },
+			{ "subsets", 2, log_uint, &subsets },
+			{ "type", 1, log_string,
+			   (void*) check_null(get_set_type(lc, rs)) },
+		};
+
+		log_fields(lc, log_handlers, ARRAY_SIZE(log_handlers));
+	} else {
+		const char *fmt[] = {
+			"name   : %s\n"
+			"size   : %" PRIu64 "\n"
+			"stride : %u\n"
+			"type   : %s\n"
+			"status : %s\n"
+			"subsets: %u\n"
+			"devs   : %u\n"
+			"spares : %u",
+			"%s",
+			"%s:%" PRIu64 ":%u:%s:%s:%u:%u:%u",
+		};
+		unsigned int o = ARRAY_LIMIT(fmt, lc_opt(lc, LC_COLUMN));
+	
+		log_print(lc, fmt[o],
+			  rs->name, sectors, rs->stride,
+			  check_null(get_set_type(lc, rs)),
+			  check_null(get_status(lc, rs->status)),
+			  subsets, devs, spares);
+	
+	}
+
+	if (OPT_COLUMN(lc) > 2) {
+		struct raid_dev *rd;
+	
+		list_for_each_entry(rd, &rs->devs, devs)
+			log_rd(lc, &rd->list);
+	}
+}
+
+static int group_active(struct lib_context *lc, struct raid_set *rs)
+{
+	struct raid_set *r;
+
+	list_for_each_entry(r, &rs->sets, list) {
+		if (dm_status(lc, r))
+			return 1;
+	}
+
+	return 0;
+}
+
+/* FIXME: Spock, do something better (neater). */
+void display_set(struct lib_context *lc, void *v,
+		 enum active_type active, int top)
+{
+	struct raid_set *rs = v;
+	struct raid_set *r;
+	int dmstatus = T_GROUP(rs) ? group_active(lc, rs) : dm_status(lc, rs);
+
+	if (((active & D_ACTIVE) && !dmstatus) ||
+	    ((active & D_INACTIVE) && dmstatus))
+		return;
+
+	if (!OPT_COLUMN(lc)) {
+		if (T_GROUP(rs) && !OPT_GROUP(lc))
+			log_print(lc, "*** Group superset %s", rs->name);
+		else {
+			log_print(lc, "%s %s%s%set",
+		  		top ? "-->" : "***",
+		  		S_INCONSISTENT(rs->status) ?
+				"*Inconsistent* " : "",
+		  		dm_status(lc, rs) ? "Active " : "",
+		  		SETS(rs) ? "Supers" : (top ? "Subs" : "S"));
+		}
+	}
+
+	log_rs(lc, rs);
+
+	/* Optionally display subsets. */
+	if (T_GROUP(rs) || /* Always display for GROUP sets. */
+	    OPT_SETS(lc) > 1 ||
+	    OPT_COLUMN(lc) > 2) {
+		list_for_each_entry(r, &rs->sets, list)
+			display_set(lc, r, active, ++top);
+	}
+}
+
+/*
+ * Display information about supported RAID metadata formats
+ * (ie. registered format handlers)
+ */
+static void _list_formats(struct lib_context *lc, enum fmt_type type)
+{
+	struct format_list *fmt_list;
+
+	list_for_each_entry(fmt_list, LC_FMT(lc), list) {
+		if (type == fmt_list->fmt->format)
+			log_format(lc, fmt_list->fmt);
+	}
+}
+
+int list_formats(struct lib_context *lc, int arg)
+{
+	log_info(lc, "supported metadata formats:");
+	_list_formats(lc, FMT_RAID);
+	_list_formats(lc, FMT_PARTITION);
+
+	return 1;
+}
/cvs/dm/dmraid/lib/format/README,v  -->  standard output
revision 1.1
--- dmraid/lib/format/README
+++ -	2008-02-22 16:57:40.348899000 +0000
@@ -0,0 +1,7 @@
+
+This directory is the top for class-subdrectories containing dmraid
+metadata format handlers. Itself holds the format handler core with
+the handler registry.
+
+See template/README and template/template.[ch] for a metadata format
+handler template which shall (hopefully) help to implement your own.
/cvs/dm/dmraid/lib/format/format.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/format.c
+++ -	2008-02-22 16:57:40.520505000 +0000
@@ -0,0 +1,543 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include "internal.h"
+#include "ondisk.h"
+
+/*
+ * Metadata format handler registry.
+ */
+
+/*
+ * Used for development.
+ *
+ * Comment next line out to avoid pre-registration
+ * checks on metadata format handlers.
+ */
+#define	CHECK_FORMAT_HANDLER
+#ifdef	CHECK_FORMAT_HANDLER
+/*
+ * Check that mandatory members of a metadata form handler are present.
+ * 
+ * We can only use log_err and log_print here, because debug and verbose
+ * options are checked for later during initialization...
+ */
+
+/*
+ * Because we have a bunch of members to check,
+ * let's define them as an array.
+ */
+#define offset(member)	struct_offset(dmraid_format, member)
+
+struct format_member {
+	const unsigned short offset;
+	const unsigned char all;
+	const unsigned char method;
+	const char *msg;
+} __attribute__ ((packed));
+
+static struct format_member format_member[] = {
+	{ offset(name),   1, 0, "name" },
+	{ offset(descr),  1, 0, "description" },
+	{ offset(caps),   0, 0, "capabilities" },
+	{ offset(read),   1, 1, "read" },
+	{ offset(write),  0, 1, "write" },
+	{ offset(group),  1, 1, "group" },
+	{ offset(check),  1, 1, "check" },
+	{ offset(events), 0, 0, "events array" },
+#ifdef	NATIVE_LOG
+	{ offset(log),    0, 1, "log" },
+#endif
+};
+#undef	offset
+
+static int check_member(struct lib_context *lc, struct dmraid_format *fmt,
+			struct format_member *member)
+{
+	if ((!member->all && fmt->format != FMT_RAID) ||
+	    *((unsigned long*) (((unsigned char*) fmt) + member->offset)))
+		return 0;
+
+	LOG_ERR(lc, 1, "%s: missing metadata format handler %s%s",
+		fmt->name, member->msg, member->method ? " method" : "");
+}
+
+static int check_format_handler(struct lib_context *lc,
+				struct dmraid_format *fmt)
+{
+	unsigned int error = 0;
+	struct format_member *fm = format_member;
+
+	if (!fmt)
+		BUG(lc, 0, "NULL metadata format handler");
+
+	while (fm < ARRAY_END(format_member))
+		error += check_member(lc, fmt, fm++);
+
+	return !error;
+}
+#endif /* CHECK_FORMAT_HANDLER */
+
+/*
+ * Register a RAID metadata format handler.
+ */
+int register_format_handler(struct lib_context *lc, struct dmraid_format *fmt)
+{
+	struct format_list *fl;
+
+#ifdef	CHECK_FORMAT_HANDLER
+	if (!check_format_handler(lc, fmt))
+		return 0;
+#undef	CHECK_FORMAT_HANDLER
+#endif
+	if ((fl = dbg_malloc(sizeof(*fl)))) {
+		fl->fmt = fmt;
+		list_add_tail(&fl->list, LC_FMT(lc));
+	}
+
+	return fl ? 1 : 0;
+}
+
+/*
+ * (Un)Register all format handlers.
+ *
+ * I use an array because of the growing number...
+ */
+static struct register_fh {
+	int (*func)(struct lib_context *lc);
+} register_fh[] = {
+#include "register.h"
+	{ NULL },
+};
+
+void unregister_format_handlers(struct lib_context *lc)
+{
+	struct list_head *elem, *tmp;
+
+	list_for_each_safe(elem, tmp, LC_FMT(lc)) {
+		list_del(elem);
+		dbg_free(list_entry(elem, struct format_list, list));
+	}
+}
+
+int register_format_handlers(struct lib_context *lc)
+{
+	int ret = 1;
+	struct register_fh *fh;
+
+	for (fh = register_fh; fh->func; fh++) {
+		if ((ret = fh->func(lc)))
+			continue;
+
+		/* Clean up in case of error. */
+		log_err(lc, "registering format");
+		unregister_format_handlers(lc);
+		break;
+	}
+
+	return ret;
+}
+/* END metadata format handler registry. */
+
+
+/*
+ * Other metadata format handler support functions.
+ */
+
+/* Allocate private space in format handlers (eg, for on-disk metadata). */
+void *alloc_private(struct lib_context *lc, const char *who, size_t size)
+{
+	void *ret;
+
+	if (!(ret = dbg_malloc(size)))
+		log_err(lc, "allocating %s metadata", who);
+
+	return ret;
+}
+
+/* Allocate private space in format handlers and read data off device. */
+void *alloc_private_and_read(struct lib_context *lc, const char *who,
+			     size_t size, char *path, loff_t offset)
+{
+	void *ret;
+
+	if ((ret = alloc_private(lc, who, size))) {
+		if (!read_file(lc, who, path, ret, size, offset)) {
+			dbg_free(ret);
+			ret = NULL;
+		}
+	}
+
+	return ret;
+}
+
+
+/* Allocate metadata sector array in format handlers. */
+void *alloc_meta_areas(struct lib_context *lc, struct raid_dev *rd,
+		       const char *who, unsigned int n)
+{
+	void *ret;
+
+	if ((ret = alloc_private(lc, who, n * sizeof(*rd->meta_areas))))
+		rd->areas = n;
+
+	return ret;
+}
+
+/* Simple metadata write function for format handlers. */
+static int _write_metadata(struct lib_context *lc, const char *handler,
+			   struct raid_dev *rd, int idx, int erase)
+{
+	int ret = 0;
+	void *p, *tmp;
+
+	if (idx >= rd->areas)
+		goto out;
+
+	p = tmp = rd->meta_areas[idx].area;
+	if (erase &&
+	    !(p = alloc_private(lc, handler, rd->meta_areas[idx].size)))
+		goto out;
+
+	ret = write_file(lc, handler, rd->di->path, (void*) p,
+			 rd->meta_areas[idx].size,
+			 rd->meta_areas[idx].offset << 9);
+
+	log_level(lc, ret ? _PLOG_DEBUG : _PLOG_ERR, 
+		  "writing metadata to %s, offset %" PRIu64 " sectors, "
+		  "size %zu bytes returned %d",
+		  rd->di->path, rd->meta_areas[idx].offset,
+		  rd->meta_areas[idx].size, ret); 
+
+	if (p != tmp)
+		dbg_free(p);
+
+  out:
+	return ret;
+}
+
+int write_metadata(struct lib_context *lc, const char *handler,
+		   struct raid_dev *rd, int idx, int erase)
+{
+	unsigned int i;
+
+	if (idx > -1)
+		return _write_metadata(lc, handler, rd, idx, erase);
+
+	for (i = 0; i < rd->areas; i++) {
+		if (!_write_metadata(lc, handler, rd, i, erase))
+			return 0;
+	}
+
+	return 1;
+}
+
+/*
+ * Check devices in a RAID set.
+ *
+ * a. spares in a mirror set need to be large enough.
+ * b. # of devices correct.
+ */
+static void _check_raid_set(struct lib_context *lc, struct raid_set *rs,
+		       unsigned int (*f_devices)(struct raid_dev *rd,
+						 void *context),
+		       void *f_devices_context,
+		       int (*f_check)(struct lib_context *lc,
+				      struct raid_set *rs,
+				      struct raid_dev *rd,
+				      void *context),
+		       void *f_check_context,
+		       const char *handler)
+{
+	unsigned int devs;
+	uint64_t sectors;
+	struct raid_dev *rd;
+
+	if (!DEVS(rs))
+		return;
+
+	sectors = total_sectors(lc, rs);
+	rs->total_devs = devs = count_devs(lc, rs, ct_dev);
+	list_for_each_entry(rd, &rs->devs, devs) {
+		unsigned int devices = f_devices(rd, f_devices_context);
+		/* FIXME: error if the metadatas aren't all the same? */
+		rs->found_devs = devices;
+
+		log_dbg(lc, "checking %s device \"%s\"", handler, rd->di->path);
+		if (T_SPARE(rd) &&
+		    rs->type == t_raid1 &&	/* FIXME: rs->type check ? */
+		    rd->sectors != sectors) {
+			rd->status = s_inconsistent;
+			log_err(lc, "%s: size mismatch in set \"%s\", spare "
+				"\"%s\"", handler, rs->name, rd->di->path);
+			continue;
+		}
+
+		if (devs != devices &&
+		    f_check &&
+		    !f_check(lc, rs, rd, f_check_context)) {
+			rd->status = s_broken;
+			log_err(lc, "%s: wrong # of devices in RAID "
+				"set \"%s\" [%u/%u] on %s",
+	   			handler, rs->name, devs, devices, rd->di->path);
+		} else
+			rd->status = s_ok;
+	}
+}
+
+/*
+ * Update RAID set state based on operational subsets/devices.
+ *
+ * In case of a RAID set hierachy, check availability of subsets
+ * and set superset to broken in case *all* subsets are broken.
+ * If at least one is still available, set to inconsistent.
+ *
+ * In case of lowest level RAID sets, check consistence of devices
+ * and make the above decision at the device level.
+ */
+static void _set_rs_status(struct lib_context *lc, struct raid_set *rs,
+			   unsigned int i, unsigned int operational)
+{
+	if (operational == i)
+		rs->status = s_ok;
+	else if (operational)
+		rs->status = s_inconsistent;
+	else
+		rs->status = s_broken;
+
+	log_dbg(lc, "set status of set \"%s\" to %u", rs->name, rs->status);
+}
+
+static int set_rs_status(struct lib_context *lc, struct raid_set *rs)
+{
+	unsigned int i = 0, operational = 0;
+	struct raid_set *r;
+	struct raid_dev *rd;
+
+	/* Set status of subsets. */
+	list_for_each_entry(r, &rs->sets, list) {
+		/* Check subsets to set status of superset. */
+		i++;
+		if (S_OK(r->status) || S_INCONSISTENT(r->status))
+			operational++;
+	}
+	
+	/* Check status of devices... */
+	list_for_each_entry(rd, &rs->devs, devs) {
+		i++;
+		if (S_OK(rd->status))
+			operational++;
+	}
+	
+	_set_rs_status(lc, rs, i, operational);
+
+	return S_BROKEN(rs->status) ? 0 : 1;
+}
+
+/*
+ * Check stack of RAID sets.
+ *
+ * This tiny helper function avoids coding recursive
+ * RAID set stack unrolling in every metadata format handler.
+ */
+int check_raid_set(struct lib_context *lc, struct raid_set *rs,
+		   unsigned int (*f_devices)(struct raid_dev *rd,
+					     void *context),
+		   void *f_devices_context,
+		   int (*f_check)(struct lib_context *lc, struct raid_set *r,
+				  struct raid_dev *rd, void *context),
+		   void *f_check_context,
+		   const char *handler)
+{
+	struct raid_set *r;
+
+	list_for_each_entry(r, &rs->sets, list)
+		check_raid_set(lc, r, f_devices, f_devices_context,
+			       f_check, f_check_context, handler);
+
+	/* Never check group RAID sets. */
+	if (!T_GROUP(rs))
+		_check_raid_set(lc, rs, f_devices, f_devices_context,
+				f_check, f_check_context, handler);
+
+	return set_rs_status(lc, rs);
+}
+
+/* Initialize a RAID sets type and stride. */
+int init_raid_set(struct lib_context *lc, struct raid_set *rs,
+		  struct raid_dev *rd, unsigned int stride,
+		  unsigned int type, const char *handler)
+{
+	if (T_UNDEF(rd))
+		LOG_ERR(lc, 0, "%s: RAID type %u not supported", handler, type);
+
+	if (T_SPARE(rs) || T_UNDEF(rs))
+		rs->type = rd->type;
+	else if (!T_SPARE(rd) && rs->type != rd->type)
+		log_err(lc, "%s: RAID type mismatch in \"%s\" on  %s",
+			handler, rs->name, rd->di->path);
+
+	if (rs->stride) {
+		if (rs->stride != stride)
+			LOG_ERR(lc, 0,
+				"%s: stride inconsistency detected on \"%s\"",
+				handler, rd->di->path);
+	} else
+		rs->stride = stride;
+
+	return 1;
+}
+
+/* Discover RAID metadata and setup RAID device. */
+struct raid_dev *read_raid_dev(
+	struct lib_context *lc,
+	struct dev_info *di,
+	void* (*f_read_metadata)(struct lib_context *lc, struct dev_info *di,
+				 size_t *size, uint64_t *offset,
+				 union read_info *info),
+	size_t size, uint64_t offset,
+	void (*f_to_cpu)(void *meta),
+	int (*f_is_meta)(struct lib_context *lc, struct dev_info *di,
+			 void *meta),
+	void (*f_file_metadata)(struct lib_context *lc, struct dev_info *di,
+				void *meta),
+	int (*f_setup_rd)(struct lib_context *lc, struct raid_dev *rd,
+			  struct dev_info *di, void *meta,
+			  union read_info *info),
+	const char *handler)
+{
+	struct raid_dev *rd = NULL;
+	void *meta;
+	union read_info info;
+
+	/*
+	 * In case the metadata format handler provides a special
+	 * metadata read function, use that. If not, allocate and
+	 * read size from offset.
+	 */
+	meta = f_read_metadata ? f_read_metadata(lc, di, &size, &offset,
+						 &info) :
+				 alloc_private_and_read(lc, handler, size,
+							di->path, offset);
+	if (!meta)
+		goto out;
+
+	/* If endianess conversion function provided -> call it. */
+	if (f_to_cpu)
+		f_to_cpu(meta);
+
+	/* Optionally check that metadata is valid. */
+	if (f_is_meta && !f_is_meta(lc, di, meta))
+		goto bad;
+
+	/* If metadata file function provided -> call it else default filing. */
+	if (f_file_metadata)
+		f_file_metadata(lc, di, meta);
+	else {
+		file_metadata(lc, handler, di->path, meta, size, offset);
+		file_dev_size(lc, handler, di);
+	}
+
+	/* Allocate RAID device structure. */
+	if (!(rd = alloc_raid_dev(lc, handler)))
+		goto bad;
+
+	/* Use metadata format handler setup function on it. */
+	if (f_setup_rd(lc, rd, di, meta, &info))
+		goto out;
+
+	log_err(lc, "%s: setting up RAID device %s", handler, di->path);
+	free_raid_dev(lc, &rd);
+	goto out;
+
+   bad:
+	dbg_free(meta);
+   out:
+	return rd;
+}
+
+/* Check if format identifier is valid. */
+int check_valid_format(struct lib_context *lc, char *name)
+{
+	struct format_list *fl;
+
+	/* FIXME: support wildcards. */
+	list_for_each_entry(fl, LC_FMT(lc), list) {
+		if (!strncmp(name, fl->fmt->name, strlen(name)))
+			return 1;
+	}
+
+	return 0;
+}
+
+/*
+ * Set up a format capabilities (ie, RAID levels) string array.
+ */
+const char **get_format_caps(struct lib_context *lc, struct dmraid_format *fmt)
+{
+	int i;
+	char *caps, *p;
+	const char **ret = NULL, delim = ',';
+
+	if (fmt->caps && (caps = dbg_strdup((char*) fmt->caps))) {
+		/* Count capabilities delimiters. */
+		for (i = 0, p = caps; (p = remove_delimiter(p, delim)); i++)
+			add_delimiter(&p, delim);
+
+		if ((ret = dbg_malloc(sizeof(*ret) * (i + 2)))) {
+			for (i = 0, p = caps - 1; p;
+			     (p = remove_delimiter(p, delim)))
+				ret[i++] = ++p;
+		} else
+			dbg_free(caps);
+	}
+
+	return ret;
+}
+
+void free_format_caps(struct lib_context *lc, const char **caps)
+{
+	if (caps) {
+		dbg_free((char*) *caps);
+		dbg_free(caps);
+	}
+}
+
+/*
+ * Allocate a RAID superset and link the subset to it.
+ */
+struct raid_set *join_superset(struct lib_context *lc,
+			       char *(*f_name)(struct lib_context *lc,
+					       struct raid_dev *rd,
+					       unsigned int subset),
+			       void (*f_create)(struct raid_set *super,
+					        void *private),
+			       int (*f_set_sort)(struct list_head *pos,
+						 struct list_head *new),
+			       struct raid_set *rs,
+			       struct raid_dev *rd)
+{
+	char *n;
+	struct raid_set *ret = NULL;
+
+	if ((n = f_name(lc, rd, 0))) {
+		if ((ret = find_or_alloc_raid_set(lc, n, FIND_TOP, NO_RD,
+						  LC_RS(lc), f_create, rd)) &&
+		    !find_set(lc, &ret->sets, rs->name, FIND_TOP))
+			list_add_sorted(lc, &ret->sets, &rs->list, f_set_sort);
+
+		dbg_free(n);
+	}
+
+	return ret;
+}
+
+/* Display 'zero sectors on RAID' device error. */
+int log_zero_sectors(struct lib_context *lc, char *path, const char *handler)
+{
+	LOG_ERR(lc, 0, "%s: zero sectors on %s", handler, path);
+}
/cvs/dm/dmraid/lib/format/ondisk.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ondisk.h
+++ -	2008-02-22 16:57:40.630250000 +0000
@@ -0,0 +1,25 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _ONDISK_H_
+#define	_ONDISK_H_
+
+#include "ataraid/asr.h"
+#include "ataraid/hpt37x.h"
+#include "ataraid/hpt45x.h"
+#include "ataraid/isw.h"
+#include "ataraid/jm.h"
+#include "ataraid/lsi.h"
+#include "ataraid/nv.h"
+#include "ataraid/pdc.h"
+#include "ataraid/via.h"
+#include "ataraid/sil.h"
+#include "ataraid/asr.h"
+
+#include "partition/dos.h"
+
+#endif
/cvs/dm/dmraid/lib/format/register.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/register.h
+++ -	2008-02-22 16:57:40.709286000 +0000
@@ -0,0 +1,33 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * List of format handler register functions
+ */
+
+#ifndef	_REGISTER_H_
+#define	_REGISTER_H_
+
+#define	xx(type)	{ register_ ## type },
+
+	/* Metadata format handlers. */
+	xx(asr)
+	xx(hpt37x)
+	xx(hpt45x)
+	xx(isw)
+	xx(jm)
+	xx(lsi)
+	xx(nv)
+	xx(pdc)
+	xx(sil)
+	xx(via)
+
+	/* DOS partition type handler. */
+	xx(dos)
+
+#undef	xx
+#endif
/cvs/dm/dmraid/lib/format/ataraid/README,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/README
+++ -	2008-02-22 16:57:40.787423000 +0000
@@ -0,0 +1,20 @@
+
+This directory contains metadata format handlers for various ATARAID cards.
+
+Some useful metadata offsets/lengths (in 512 byte sector sizes):
+----------------------------------------------------------------
+
+Highpoint 45x *doesn't* zero the one on-disk metadata at -11,
+length 1 on drive removal.
+
+Intel Software RAID has metadata at -2 (conditionally -n), length in
+metadata structure member mpbSize.
+
+LSI Logic MegaRAID has metadata at -1, length 1.
+
+NVidea NForce has metadata at -2, length 1.
+
+Silicon Image Medley zeroes all 4 on-disk metadata areas
+(at -1, -512, -1024, -1536; length 4) on drive removal.
+
+VIA has metadata at -1, length 1.
/cvs/dm/dmraid/lib/format/ataraid/asr.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/asr.c
+++ -	2008-02-22 16:57:40.867817000 +0000
@@ -0,0 +1,873 @@
+/*
+ * Adaptec HostRAID ASR format interpreter for dmraid.
+ * Copyright (C) 2005-2006 IBM, All rights reserved.
+ * Written by Darrick Wong <djwong at us.ibm.com>
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <errno.h>
+#include <netinet/in.h>
+
+#define	HANDLER	"asr"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "asr.h"
+
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+#define SPARE_ARRAY	".asr_spares"
+
+static int asr_write(struct lib_context *lc,  struct raid_dev *rd, int erase);
+
+/* Map ASR disk status to dmraid status */
+static enum status disk_status(struct asr_raid_configline *disk) {
+	if (disk == NULL)
+		return s_undef;
+
+	switch (disk->raidstate) {
+	case LSU_COMPONENT_STATE_OPTIMAL:
+		return s_ok;
+
+	case LSU_COMPONENT_STATE_DEGRADED:
+	case LSU_COMPONENT_STATE_FAILED:
+		return s_broken;
+
+	case LSU_COMPONENT_STATE_UNINITIALIZED:
+	case LSU_COMPONENT_STATE_UNCONFIGURED:
+		return s_inconsistent;
+
+	case LSU_COMPONENT_SUBSTATE_BUILDING:
+	case LSU_COMPONENT_SUBSTATE_REBUILDING:
+	case LSU_COMPONENT_STATE_REPLACED:
+		return s_nosync;
+
+	default:
+		return s_undef;
+	}
+}
+		
+/* Extract config line from metadata */
+static struct asr_raid_configline *get_config(struct asr *asr, uint32_t magic)
+{
+	unsigned int i;
+	
+	for (i = 0; i < asr->rt->elmcnt; i++) {
+		if (asr->rt->ent[i].raidmagic == magic)
+			return asr->rt->ent + i;
+	}
+
+	return NULL;
+}
+
+/* Get this disk's configuration */
+static struct asr_raid_configline *this_disk(struct asr *asr)
+{
+	return get_config(asr, asr->rb.drivemagic);
+}
+
+/* Make up RAID device name. */
+static size_t _name(struct lib_context *lc, struct asr *asr, char *str,
+		    size_t len)
+{
+	struct asr_raid_configline *cl = this_disk(asr);
+
+	if (cl)
+		return snprintf(str, len, "%s_%s", HANDLER, cl->name);
+
+	LOG_ERR(lc, 0, "%s: Could not find device in config table!", handler);
+}
+
+/* Figure out a name for the RAID device. */
+static char *name(struct lib_context *lc, struct asr *asr)
+{
+	size_t len;
+	char *ret;
+
+	if ((ret = dbg_malloc((len = _name(lc, asr, NULL, 0) + 1)))) {
+		_name(lc, asr, ret, len);
+		/* Why do we call mk_alpha?  This makes labels like
+		 * "OS-u320-15k" become "OS-udca-bek", which is confusing.
+		 * mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN); */
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/* Stride size */
+static inline unsigned int stride(struct asr_raid_configline *cl)
+{
+	return cl ? cl->strpsize: 0;
+}
+
+/* Mapping of template types to generic types */
+/*
+ * FIXME: This needs more examination.  Does HostRAID do linear
+ * combination?  The BIOS implies that it only does RAID 0, 1 and 10.
+ * The emd driver implied support for RAID3/4/5, but dm doesn't
+ * do any of those right now (RAID4 and RAID5 are in the works).
+ */
+static struct types types[] = {
+	{ ASR_RAID0,   t_raid0 },
+	{ ASR_RAID1,   t_raid1 },
+	{ ASR_RAIDSPR, t_spare },
+        { 0, t_undef}
+};
+
+/* Map the ASR raid type codes into dmraid type codes. */
+static enum type type(struct asr_raid_configline *cl)
+{
+	return cl ? rd_type(types, (unsigned int) cl->raidtype) : t_undef;
+}
+
+/*
+ * Read an ASR RAID device.  Fields are big endian, so
+ * need to convert them if we're on a LE machine (i386, etc).
+ */
+#define ASR_BLOCK	0x01
+#define ASR_TABLE	0x02
+#define ASR_EXTTABLE 	0x04
+
+#if	BYTE_ORDER == LITTLE_ENDIAN
+static void cvt_configline(struct asr_raid_configline *cl)
+{
+	CVT16(cl->raidcnt);
+	CVT16(cl->raidseq);
+	CVT32(cl->raidmagic);
+	CVT32(cl->raidid);
+	CVT32(cl->loffset);
+	CVT32(cl->lcapcty);
+	CVT16(cl->strpsize);
+	CVT16(cl->biosInfo);
+	CVT32(cl->lsu);
+	CVT16(cl->blockStorageTid);
+	CVT32(cl->curAppBlock);
+	CVT32(cl->appBurstCount);
+}
+
+static void to_cpu(void *meta, unsigned int cvt)
+{
+	int i;
+	struct asr *asr = meta;
+	int elmcnt = asr->rt->elmcnt;
+
+	int use_old_elmcnt = (asr->rt->ridcode == RVALID2);
+
+	if (cvt & ASR_BLOCK) {
+		CVT32(asr->rb.b0idcode);
+		CVT16(asr->rb.biosInfo);
+		CVT32(asr->rb.fstrsvrb);
+		CVT16(asr->rb.svBlockStorageTid);
+		CVT16(asr->rb.svtid);
+		CVT32(asr->rb.drivemagic);
+		CVT32(asr->rb.fwTestMagic);
+		CVT32(asr->rb.fwTestSeqNum);
+		CVT32(asr->rb.smagic);
+		CVT32(asr->rb.raidtbl);
+	}
+
+	if (cvt & ASR_TABLE) {
+		CVT32(asr->rt->ridcode);
+		CVT32(asr->rt->rversion);
+		CVT16(asr->rt->maxelm);
+		CVT16(asr->rt->elmcnt);
+		if (!use_old_elmcnt)
+			elmcnt = asr->rt->elmcnt;
+		CVT16(asr->rt->elmsize);
+		CVT32(asr->rt->raidFlags);
+		CVT32(asr->rt->timestamp);
+		CVT16(asr->rt->rchksum);
+		CVT32(asr->rt->sparedrivemagic);
+		CVT32(asr->rt->raidmagic);
+		CVT32(asr->rt->verifyDate);
+		CVT32(asr->rt->recreateDate);
+
+		/* Convert the first seven config lines */
+		for (i = 0; i < (elmcnt < 7 ? elmcnt : 7); i++) 
+			cvt_configline(asr->rt->ent + i);
+		
+	}
+
+	if (cvt & ASR_EXTTABLE) {
+		for (i = 7; i < elmcnt; i++) {
+			cvt_configline(asr->rt->ent + i);
+		}
+	}
+}
+
+#else
+# define to_cpu(x, y)
+#endif
+
+/* Compute the checksum of RAID metadata */
+static unsigned int compute_checksum(struct asr *asr)
+{
+	uint8_t *ptr;
+	unsigned int i, checksum;
+
+	/* Compute checksum. */
+	ptr = (uint8_t*) asr->rt->ent;
+	checksum = 0;
+	for (i = 0; i < sizeof(*asr->rt->ent) * asr->rt->elmcnt; i++)
+		checksum += ptr[i];
+
+	return checksum & 0xFFFF;
+}
+
+/* Read extended metadata areas */
+static int read_extended(struct lib_context *lc, struct dev_info *di,
+			 struct asr *asr)
+{
+	unsigned int remaining, i, chk;
+	int j;
+
+	log_info(lc, "%s: reading extended data", di->path);
+	
+	/* Read the RAID table. */
+	if (!read_file(lc, handler, di->path, asr->rt, ASR_DISK_BLOCK_SIZE,
+		       (uint64_t) asr->rb.raidtbl * ASR_DISK_BLOCK_SIZE))
+		LOG_ERR(lc, 0, "%s: Could not read metadata.", handler);
+
+	/* Convert it */
+	to_cpu(asr, ASR_TABLE);
+	
+	/* Is this ok? */
+	if (asr->rt->ridcode != RVALID2)
+		LOG_ERR(lc, 0, "%s: Invalid magic number in RAID table; "
+			"saw 0x%X, expected 0x%X.", handler, asr->rt->ridcode,
+			RVALID2);
+
+	/* Have we a valid element count? */
+	if (asr->rt->elmcnt >= asr->rt->maxelm)
+		LOG_ERR(lc, 0, "%s: Invalid RAID config table count.\n",
+			handler);
+
+	/* Is each element the right size? */
+	if (asr->rt->elmsize != sizeof(struct asr_raid_configline))
+		LOG_ERR(lc, 0, "%s: RAID config line is the wrong size.\n",
+			handler);
+
+	/* Figure out how much else we need to read. */
+	if (asr->rt->elmcnt > 7) {
+		remaining = asr->rt->elmsize * (asr->rt->elmcnt - 7);
+		if (!read_file(lc, handler, di->path, asr->rt->ent + 7,
+			       remaining, (uint64_t)(asr->rb.raidtbl + 1) *
+			       ASR_DISK_BLOCK_SIZE))
+			return 0;
+
+		to_cpu(asr, ASR_EXTTABLE);
+	}
+
+	chk = compute_checksum(asr);
+	if (chk != asr->rt->rchksum)
+		LOG_ERR(lc, 0,"%s: Invalid RAID config table checksum "
+			       "(0x%X vs. 0x%X).",
+			handler, chk, asr->rt->rchksum);
+	
+	/* Process the name of each line of the config line. */
+	for (i = 0; i < asr->rt->elmcnt; i++) {
+		/* 
+		 * Weird quirks of the name field of the config line:
+		 *
+		 * - SATA HostRAID w/ ICH5 on IBM x226: The name field is null
+		 *   in the drive config lines.  The zeroeth item does have a
+		 *   name, however.
+		 * - Spares on SCSI HostRAID on IBM x226: The name field for
+		 *   all config lines is null.
+		 * 
+		 * So, we'll assume that we can copy the name from the zeroeth
+		 * element in the array.  The twisted logic doesn't seem to
+		 * have a problem with either of the above cases, though
+		 * attaching spares is going to be a tad tricky (primarily
+		 * because there doesn't seem to be a way to attach a spare to
+		 * a particular array; presumably the binary driver knows how
+		 * or just grabs a disk out of the spare pool.
+		 *
+		 * (Yes, the binary driver _does_ just grab a disk from the
+		 * global spare pool.  We must teach dm about this...?)
+		 *
+		 * This is nuts.
+		 */
+		if (!asr->rt->ent[i].name)
+			memcpy(asr->rt->ent[i].name, asr->rt->ent[0].name, 16);
+
+		/* Now truncate trailing whitespace in the name. */
+		for (j = 15; j >= 0; j--) {
+			if (asr->rt->ent[i].name[j] != ' ')
+				break;
+		}
+		asr->rt->ent[i].name[j + 1] = 0;
+	}
+
+	return 1;
+}
+
+static int is_asr(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct asr *asr = meta;
+
+	/*
+	 * Check our magic numbers and that the version == v8.
+	 * We don't support anything other than that right now.
+	 */
+	if (asr->rb.b0idcode == B0RESRVD &&
+	    asr->rb.smagic == SVALID) {
+		if (asr->rb.resver == RBLOCK_VER)
+			return 1;
+		
+		LOG_ERR(lc, 0,
+			"%s: ASR v%d detected, but we only support v8.\n",
+			handler, asr->rb.resver);
+	}
+
+	return 0;
+}
+
+/*
+ * Attempt to interpret ASR metadata from a block device.  This function
+ * returns either NULL (not an ASR) or a pointer to a descriptor struct.
+ * Note that the struct should be fully converted to the correct endianness
+ * by the time this function returns.
+ *
+ * WARNING: If you take disks out of an ASR HostRAID array and plug them in
+ * to a normal SCSI controller, the array will still show up!  Even if you
+ * scribble over the disks!  I assume that the a320raid binary driver only
+ * does its HostRAID magic if your controller is in RAID mode... but dmraid
+ * lacks this sort of visibility as to where its block devices come from.
+ * This is EXTREMELY DANGEROUS if you aren't careful!
+ */
+static void *read_metadata_areas(struct lib_context *lc, struct dev_info *di,
+				 size_t *sz, uint64_t *offset,
+				 union read_info *info)
+{
+	size_t size = ASR_DISK_BLOCK_SIZE;
+	uint64_t asr_sboffset = ASR_CONFIGOFFSET;
+	struct asr *asr;
+	struct asr_raid_configline *cl;
+
+	/*
+	 * Read the ASR reserved block on each disk.  This is the very
+	 * last sector of the disk, and we're really only interested in
+	 * the two magic numbers, the version, and the pointer to the
+	 * RAID table.  Everything else appears to be unused in v8.
+	 */
+	if (!(asr = alloc_private(lc, handler, sizeof(struct asr))))
+		goto bad0;
+	
+	if (!(asr->rt = alloc_private(lc, handler, sizeof(struct asr_raidtable))))
+		goto bad1;
+
+	if (!read_file(lc, handler, di->path, &asr->rb, size, asr_sboffset))
+		goto bad2;
+
+	/*
+	 * Convert metadata and read in 
+	 */
+	to_cpu(asr, ASR_BLOCK);
+
+	/* Check Signature and read optional extended metadata. */
+	if (!is_asr(lc, di, asr) ||
+	    !read_extended(lc, di, asr))
+		goto bad2;
+
+	/*
+	 * Now that we made sure that we have all the metadata, we exit.
+	 */
+	cl = this_disk(asr);
+	if (cl->raidstate == LSU_COMPONENT_STATE_FAILED)
+		goto bad2;
+
+	goto out;
+
+   bad2:
+	dbg_free(asr->rt);
+   bad1:
+	asr->rt = NULL;
+	dbg_free(asr);
+   bad0:
+	asr = NULL;
+
+   out:
+	return (void*) asr;
+}
+
+/*
+ * "File the metadata areas" -- I think this function is supposed to declare
+ * which parts of the drive are metadata and thus off-limits to dmraid.
+ */
+static void file_metadata_areas(struct lib_context *lc, struct dev_info *di,
+				void *meta)
+{
+	struct asr *asr = meta;
+
+	/* Register the raid tables. */
+	file_metadata(lc, handler, di->path, asr->rt,
+		      ASR_DISK_BLOCK_SIZE * 17,
+		      (uint64_t)asr->rb.raidtbl * ASR_DISK_BLOCK_SIZE);
+
+	/* Record the device size if -D was specified. */
+	file_dev_size(lc, handler, di);
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+
+static struct raid_dev *asr_read(struct lib_context *lc,
+					struct dev_info *di)
+{
+	/*
+	 * NOTE: Everything called after read_metadata_areas assumes that
+	 * the reserved block, raid table and config table have been
+	 * converted to the appropriate endianness.
+	 */
+	return read_raid_dev(lc, di, read_metadata_areas, 0, 0, NULL, NULL,
+			     file_metadata_areas, setup_rd, handler);
+}
+
+static int set_sort(struct list_head *dont, struct list_head *care)
+{
+	return 0;
+}
+
+/*
+ * Compose a 64-bit ID for device sorting.
+ * Is hba:ch:lun:id ok?  It seems to be the way the binary driver
+ * does it...
+ */
+static inline uint64_t compose_id(struct asr_raid_configline *cl)
+{
+	return    ((uint64_t) cl->raidhba  << 48)
+		| ((uint64_t) cl->raidchnl << 40)
+		| ((uint64_t) cl->raidlun  << 32)
+		| (uint64_t) cl->raidid;
+}
+
+/* Sort ASR devices by for a RAID set. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return compose_id(this_disk(META(RD(new), asr))) <
+	       compose_id(this_disk(META(RD(pos), asr)));
+}
+
+/*
+ * Find the top-level RAID set for an ASR context.
+ */
+static int find_toplevel(struct lib_context *lc, struct asr *asr)
+{
+	int i, toplevel = -1;
+
+	for (i = 0; i < asr->rt->elmcnt; i++) {
+		if (asr->rt->ent[i].raidlevel == FWL)
+		{
+			toplevel = i;
+		}
+		else if (asr->rt->ent[i].raidlevel == FWL_2)
+		{
+			toplevel = i;
+			break;
+		}
+	}
+	
+	return toplevel;
+}
+
+/*
+ * Find the logical drive configuration that goes with this
+ * physical disk configuration.
+ */
+static struct asr_raid_configline *find_logical(struct asr *asr)
+{
+	int i, j;
+
+	/* This MUST be done backwards! */
+	for (i = asr->rt->elmcnt - 1; i > -1; i--) {
+		if (asr->rt->ent[i].raidmagic == asr->rb.drivemagic)
+		{
+			for (j = i - 1; j > -1; j--) {
+				if (asr->rt->ent[j].raidlevel == FWL)
+				{
+					return &asr->rt->ent[j];
+				}
+			}
+		}
+	}
+
+	return NULL;
+}
+
+/* Wrapper for name() */
+static char *js_name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	return name(lc, META(rd, asr));
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct asr *asr = META(rd, asr);
+	struct asr_raid_configline *cl = this_disk(asr);
+	struct asr_raid_configline *fwl = find_logical(asr);
+
+	/* Ignore if we've already marked this disk broken(?) */
+	if (rd->status & s_broken)
+		return 0;
+	
+	log_err(lc, "I/O error on device %s at sector %lu.",
+		e_io->rd->di->path, e_io->sector);
+
+	/* Mark the array as degraded and the disk as failed. */
+	rd->status = s_broken;
+	cl->raidstate = LSU_COMPONENT_STATE_FAILED;
+	fwl->raidstate = LSU_COMPONENT_STATE_DEGRADED;
+	/* FIXME: Do we have to mark a parent too? */
+
+	/* Indicate that this is indeed a failure. */
+	return 1;
+}
+
+/* 
+ * Add an ASR device to a RAID set.  This involves finding the raid set to
+ * which this disk belongs, and then attaching it.  Note that there are other
+ * complications, such as two-layer arrays (RAID10).
+ */
+#define BUFSIZE 128
+static struct raid_set *asr_group(struct lib_context *lc, struct raid_dev *rd)
+{
+	int top_idx;
+	struct asr *asr = META(rd, asr);
+	struct asr_raid_configline *cl = this_disk(asr);
+	struct asr_raid_configline *fwl;
+	struct raid_set *set, *sset;
+	char buf[BUFSIZE];
+
+	if (T_SPARE(rd)) {
+		/*
+		 * If this drive really _is_ attached to a specific
+		 * RAID set, then just attach it.  Really old HostRAID cards
+		 * do this... but I don't have any hardware to test this.
+		 */
+		/*
+		 * FIXME: dmraid ignores spares attached to RAID arrays.
+		 * For now, we'll let it get sucked into the ASR spare pool. 
+		 * If we need it, we'll reconfigure it; if not, nobody touches
+		 * it.
+		 *
+		set = find_set(lc, name(lc, asr), FIND_TOP, rd, LC_RS(lc),
+			       NO_CREATE, NO_CREATE_ARG);
+		 */
+
+		/* Otherwise, make a global spare pool. */
+		set = find_or_alloc_raid_set(lc, (char*)SPARE_ARRAY,
+			FIND_TOP, rd, LC_RS(lc), NO_CREATE, NO_CREATE_ARG);
+
+		/*
+		 * Setting the type to t_spare guarantees that dmraid won't
+		 * try to set up a real device-mapper mapping.
+		 */
+		set->type = t_spare;
+
+		/* Add the disk to the set. */
+		list_add_sorted(lc, &set->devs, &rd->devs, dev_sort);
+		return set;
+	}
+
+	/* Find the top level FWL/FWL2 for this device. */
+	top_idx = find_toplevel(lc, asr);
+	if (top_idx < 0) {
+		LOG_ERR(lc, NULL, "Can't find a logical array config "
+			"for disk %x\n",
+			asr->rb.drivemagic);
+	}
+
+	/* This is a simple RAID0/1 array.  Find the set. */
+	if (asr->rt->ent[top_idx].raidlevel == FWL)
+	{
+		set = find_or_alloc_raid_set(lc, name(lc, asr),
+			FIND_TOP, rd, LC_RS(lc), NO_CREATE, NO_CREATE_ARG);
+
+		set->stride = stride(cl);
+		set->status = s_ok;
+		set->type = type(find_logical(asr));
+
+		/* Add the disk to the set. */
+		list_add_sorted(lc, &set->devs, &rd->devs, dev_sort);
+		
+		return set;
+	}
+
+	/*
+	 * This is a two-level RAID array.  Attach the disk to the disk's
+	 * parent set; create it if necessary.  Then, find the top-level set
+	 * and use join_superset to attach the parent set to the top set.
+	 */
+	if (asr->rt->ent[top_idx].raidlevel == FWL_2)
+	{
+		/* First compute the name of the disk's direct parent. */
+		fwl = find_logical(asr);
+		snprintf(buf, BUFSIZE, ".asr_%s_%x_donotuse",
+			 fwl->name, fwl->raidmagic);
+		
+		/* Now find said parent. */
+		set = find_or_alloc_raid_set(lc, buf,
+			FIND_ALL, rd, NO_LIST, NO_CREATE, NO_CREATE_ARG);
+
+		if (!set)
+			LOG_ERR(lc, NULL, "Error creating RAID set.\n");
+
+		set->stride = stride(cl);
+		set->status = s_ok;
+		set->type = type(fwl);
+
+		/* Add the disk to the set. */
+		list_add_sorted(lc, &set->devs, &rd->devs, dev_sort);
+		
+		/* Find the top level set. */
+		sset = join_superset(lc, js_name, NO_CREATE,
+				     set_sort, set, rd);
+
+		if (!sset)
+			LOG_ERR(lc, NULL, "Error creating top RAID set.\n");
+
+		sset->stride = stride(cl);
+		sset->status = s_ok;
+		sset->type = type(&asr->rt->ent[top_idx]);
+
+		return sset;
+	}
+
+	/* If we land here, something's seriously wrong. */
+	LOG_ERR(lc, NULL, "Top level array config is not FWL/FWL2?\n");
+}
+
+/* Write metadata. */
+static int asr_write(struct lib_context *lc,  struct raid_dev *rd, int erase)
+{
+	int ret, i, j;
+        struct asr *asr = META(rd, asr);
+	int elmcnt = asr->rt->elmcnt;
+
+	/* Untruncate trailing whitespace in the name. */
+	for (i = 0; i < elmcnt; i++) {
+		for (j = 15; j >= 0; j--) {
+			if (asr->rt->ent[i].name[j] == 0)
+				break;
+		}
+		asr->rt->ent[i].name[j] = ' ';
+	}
+
+	/* Compute checksum */
+	asr->rt->rchksum = compute_checksum(asr);
+
+	/* Convert back to disk format */
+        to_disk(asr, ASR_BLOCK | ASR_TABLE | ASR_EXTTABLE);
+
+	/* Write data */
+        ret = write_metadata(lc, handler, rd, -1, erase);
+	
+	/* Go back to CPU format */
+        to_cpu(asr, ASR_BLOCK | ASR_TABLE | ASR_EXTTABLE);
+ 
+	/* Truncate trailing whitespace in the name. */
+	for (i = 0; i < elmcnt; i++) {
+		for (j = 15; j >= 0; j--) {
+			if (asr->rt->ent[i].name[j] != ' ')
+				break;
+		}
+		asr->rt->ent[i].name[j + 1] = 0;
+	}
+
+        return ret;
+}
+
+/*
+ * Check integrity of a RAID set.
+ */
+
+/* Retrieve the number of devices that should be in this set. */
+static unsigned int device_count(struct raid_dev *rd, void *context)
+{
+	/* Get the logical drive */
+	struct asr_raid_configline *cl = find_logical(META(rd, asr));
+	return (cl ? cl->raidcnt : 0);
+}
+
+/* Check a RAID device */
+static int check_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_dev *rd, void *context)
+{
+	/* FIXME: Assume non-broken means ok. */
+	return (rd->type != s_broken);
+}
+
+/* Start the recursive RAID set check. */
+static int asr_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, device_count, NULL, check_rd,
+			      NULL, handler);
+}
+
+static struct event_handlers asr_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+/* Dump a reserved block */
+static void dump_rb(struct lib_context *lc, struct asr_reservedblock *rb)
+{
+	DP("block magic:\t\t0x%X", rb, rb->b0idcode);
+	DP("sb0flags:\t\t\t0x%X", rb, rb->sb0flags);
+	DP("jbodEnable:\t\t%d", rb, rb->jbodEnable);
+	DP("biosInfo:\t\t\t0x%X", rb, rb->biosInfo);
+	DP("drivemagic:\t\t0x%X", rb, rb->drivemagic);
+	DP("svBlockStorageTid:\t0x%X", rb, rb->svBlockStorageTid);
+	DP("svtid:\t\t\t0x%X", rb, rb->svtid);
+	DP("resver:\t\t\t%d", rb, rb->resver);
+	DP("smagic:\t\t\t0x%X", rb, rb->smagic);
+	DP("raidtbl @ sector:\t\t%d", rb, rb->raidtbl);
+}
+
+/* Dump a raid config line */
+static void dump_cl(struct lib_context *lc, struct asr_raid_configline *cl)
+{
+	DP("config ID:\t\t0x%X", cl, cl->raidmagic);
+	DP("  name:\t\t\t\"%s\"", cl, cl->name);
+	DP("  raidcount:\t\t%d", cl, cl->raidcnt);
+	DP("  sequence #:\t\t%d", cl, cl->raidseq);
+	DP("  level:\t\t\t%d", cl, cl->raidlevel);
+	DP("  type:\t\t\t%d", cl, cl->raidtype);
+	DP("  state:\t\t\t%d", cl, cl->raidstate);
+	DP("  flags:\t\t\t0x%X", cl, cl->flags);
+	DP("  refcount:\t\t%d", cl, cl->refcnt);
+	DP("  hba:\t\t\t%d", cl, cl->raidhba);
+	DP("  channel:\t\t%d", cl, cl->raidchnl);
+	DP("  lun:\t\t\t%d", cl, cl->raidlun);
+	DP("  id:\t\t\t%d", cl, cl->raidid);
+	DP("  offset:\t\t\t%d", cl, cl->loffset);
+	DP("  capacity:\t\t%d", cl, cl->lcapcty);
+	P("  stripe size:\t\t%d KB",
+	  cl, cl->strpsize, cl->strpsize * ASR_DISK_BLOCK_SIZE / 1024);
+	DP("  BIOS info:\t\t%d", cl, cl->biosInfo);
+	DP("  phys/log lun:\t\t%d", cl, cl->lsu);
+	DP("  addedDrives:\t\t%d", cl, cl->addedDrives);
+	DP("  appSleepRate:\t\t%d", cl, cl->appSleepRate);
+	DP("  blockStorageTid:\t%d", cl, cl->blockStorageTid);
+	DP("  curAppBlock:\t\t%d", cl, cl->curAppBlock);
+	DP("  appBurstCount:\t\t%d", cl, cl->appBurstCount);
+}
+
+/* Dump a raid config table */
+static void dump_rt(struct lib_context *lc, struct asr_raidtable *rt)
+{
+	unsigned int i;
+
+	DP("ridcode:\t\t\t0x%X", rt, rt->ridcode);
+	DP("table ver:\t\t%d", rt, rt->rversion);
+	DP("max configs:\t\t%d", rt, rt->maxelm);
+	DP("configs:\t\t\t%d", rt, rt->elmcnt);
+	DP("config sz:\t\t%d", rt, rt->elmsize);
+	DP("checksum:\t\t\t0x%X", rt, rt->rchksum);
+	DP("raid flags:\t\t0x%X", rt, rt->raidFlags);
+	DP("timestamp:\t\t0x%X", rt, rt->timestamp);
+	P("irocFlags:\t\t%X%s", rt, rt->irocFlags, rt->irocFlags,
+	  rt->irocFlags & ASR_IF_BOOTABLE ? " (bootable)" : "");
+	DP("dirt, rty:\t\t%d", rt, rt->dirty);
+	DP("action prio:\t\t%d", rt, rt->actionPriority);
+	DP("spareid:\t\t\t%d", rt, rt->spareid);
+	DP("sparedrivemagic:\t\t0x%X", rt, rt->sparedrivemagic);
+	DP("raidmagic:\t\t0x%X", rt, rt->raidmagic);
+	DP("verifydate:\t\t0x%X", rt, rt->verifyDate);
+	DP("recreatedate:\t\t0x%X", rt, rt->recreateDate);
+
+	log_print(lc, "\nRAID config table:");
+	for (i = 0; i < rt->elmcnt; i++)
+		dump_cl(lc, &rt->ent[i]);
+}
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about the RAID device.
+ */
+static void asr_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	struct asr *asr = META(rd, asr);
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	dump_rb(lc, &asr->rb);
+	dump_rt(lc, asr->rt);
+}
+#endif
+
+static struct dmraid_format asr_format = {
+	.name	= HANDLER,
+	.descr	= "Adaptec HostRAID ASR",
+	.caps	= "0,1,10",
+	.format = FMT_RAID,
+	.read	= asr_read,
+	.write	= asr_write,
+	.group	= asr_group,
+	.check	= asr_check,
+	.events	= &asr_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= asr_log,
+#endif
+};
+
+/* Register this format handler with the format core */
+int register_asr(struct lib_context *lc)
+{
+	return register_format_handler(lc, &asr_format);
+}
+
+/*
+ * Set up a RAID device from what we've assembled out of the metadata.
+ */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct asr *asr = meta;
+	struct meta_areas *ma;
+	struct asr_raid_configline *cl = this_disk(asr);
+
+	if (!cl)
+		LOG_ERR(lc, 0, "%s: Could not find current disk!\n",
+			handler);		
+
+	/* We need two metadata areas */
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 2)))
+		return 0;
+
+	/* First area: raid reserved block. */
+	ma = rd->meta_areas;
+	ma->offset = ASR_CONFIGOFFSET >> 9;
+	ma->size = ASR_DISK_BLOCK_SIZE;
+	ma->area = (void*) asr;
+
+	/* Second area: raid table. */
+	ma++;
+	ma->offset = asr->rb.raidtbl;
+	ma->size = ASR_DISK_BLOCK_SIZE * 16;
+	ma->area = (void*) asr->rt;
+
+	/* Now set up the rest of the metadata info */
+        rd->di = di;
+	rd->fmt = &asr_format;
+
+	rd->status = disk_status(cl);
+	rd->type   = type(cl);
+
+	rd->offset = ASR_DATAOFFSET;
+	rd->sectors = cl->lcapcty;
+
+	return (rd->name = name(lc, asr)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/asr.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/asr.h
+++ -	2008-02-22 16:57:40.952303000 +0000
@@ -0,0 +1,388 @@
+/*
+ * Copyright (c) 2005-2006 IBM (actual code changes by Darrick Wong)
+ * 
+ * Copyright (c) 2001, 2002, 2004 Adaptec Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions, and the following disclaimer,
+ *    without modification.
+ * 2. Redistributions in binary form must reproduce at minimum a disclaimer
+ *    substantially similar to the "NO WARRANTY" disclaimer below
+ *    ("Disclaimer") and any redistribution must be conditioned upon
+ *    including a substantially similar Disclaimer requirement for further
+ *    binary redistribution.
+ * 3. Neither the names of the above-listed copyright holders nor the names
+ *    of any contributors may be used to endorse or promote products derived
+ *    from this software without specific prior written permission.
+ *
+ * Alternatively, this software may be distributed under the terms of the
+ * GNU General Public License ("GPL") version 2 as published by the Free
+ * Software Foundation.
+ *
+ * NO WARRANTY
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+ * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
+ * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGES.
+ */
+
+#ifndef _ASR_H
+#define _ASR_H
+
+/* Beginning of stuff that Darrick Wong added */
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+/* ASR metadata offset in bytes */
+#define	ASR_CONFIGOFFSET	((di->sectors - 1) << 9)
+
+/* Data offset in sectors */
+#define	ASR_DATAOFFSET		0
+
+/* Assume block size is 512.  So much for CD dmraid... */
+#define ASR_DISK_BLOCK_SIZE	512
+
+/* End of stuff that Darrick Wong added */
+
+/* Begining of stuff copied verbatim from Adaptec's emd driver. */
+
+/*
+ * This is a metadata versioning mechanism, but rather a versioning
+ * mechansim for Adaptec-sponsored changes to md.  (i.e. more of a driver
+ * version)
+ */
+#define	MD_ADAPTEC_MAJOR_VERSION	0
+#define	MD_ADAPTEC_MINOR_VERSION	0
+#define	MD_ADAPTEC_PATCHLEVEL_VERSION	13
+
+#define FW_RESERVED_BLOCKS  0x800
+
+#define MAX_SLEEPRATE_ENTRIES 10
+
+/* define lsu levels */
+#define LSU_LEVEL_PHYSICAL	1	/* Firmware Physical */
+#define LSU_LEVEL_LOGICAL	2	/* Firmware Logical */
+
+/* define RAID drive substates */
+#define FWP		0	/* Firmware Physical */
+#define FWL		1	/* Firmware Logical */
+#define OSI		2	/* Operating System Intermediate */
+#define FWL_2		3	/* Dual Level */
+
+#define ASR_RAID0		0
+#define ASR_RAID1		1
+#define ASR_RAID4		4
+#define ASR_RAID5		5
+#define ASR_RAIDRED		0xFF
+#define ASR_RAIDSPR		0xFE
+
+/*** RAID CONFIGURATION TABLE STRUCTURE ***/
+
+#define RVALID2			0x900765C4   /* Version 2+ RAID table ID code
+						signature */
+#define RCTBL_MAX_ENTRIES	127
+#define HBA_RCTBL_MAX_ENTRIES	255
+#define RTBLBLOCKS		16	     /* Size of drive's raid table
+						in blocks */
+
+/* flag bits */
+#define RCTBLCHNG					0x80	/* Set on comp OR log (NOT AND) if tbl updates needed */
+#define	COPYDIR						0x40
+#define	MIRCOPY						0x20
+#define INITIAL_BUILD_COMPLETED				0x10
+#define SMART_DISABLED					0x08
+#define WRITEBACK					0x04
+#define PREDICTIVE_ENABLE				0x02
+#define RAID_ENTRY_FLAGS_ALARM_OFF_M			0x01
+
+struct asr_raid_configline
+{
+	uint16_t	raidcnt;    /* Component count of an OSL/FWL array */
+	uint16_t	raidseq;    /* Sequence # of component to look for */
+	uint32_t	raidmagic;  /* Magic # of component to look for */
+	uint8_t		raidlevel;  /* Array level = OSL/FWL/OSI/FWP */
+	uint8_t		raidtype;   /* Array type = RAID0/1/3/5, RAIDRED,
+				       RAIDSPR */
+	uint8_t		raidstate;  /* State of logical or physical drive */
+
+	uint8_t		flags;      /* misc flags set bit positions above */
+
+	uint8_t		refcnt;     /* Number of references to this log entry */
+	uint8_t		raidhba;    /* -- not used -- Host bus adapter number
+				       or RAIDID */
+	uint8_t		raidchnl;   /* Channel number */
+	uint8_t		raidlun;    /* SCSI LUN of log/phys drv */
+	uint32_t	raidid;     /* SCSI ID of log/phys drv */
+	uint32_t	loffset;    /* Offset of data for this comp in the
+				       array */
+	uint32_t	lcapcty;    /* Capacity of log drv or space used on
+				       phys */
+	uint16_t	strpsize;   /* Stripe size in blocks of this drive */
+	uint16_t	biosInfo;   /* bios info - set by
+				       I2O_EXEC_BIOS_INFO_SET */
+	uint32_t	lsu;        /* Pointer to phys/log lun of this entry */
+	uint8_t		addedDrives;
+	uint8_t		appSleepRate;
+	uint16_t	blockStorageTid;
+	uint32_t	curAppBlock;
+	uint32_t	appBurstCount;
+	uint8_t		name[16];   /* Full name of the array. */
+} __attribute__ ((packed));
+
+struct asr_raidtable
+{
+/* raid Flag defines 32 bits 0 - FFFFFFFF */
+#define RAID_FLAGS_ALARM_OFF_M 0x00000001
+	uint32_t	ridcode;	/* RAID table signature - 0x900765C4 */
+	uint32_t	rversion;	/* Version of the RAID config table */
+	uint16_t	maxelm;		/* Maximum number of elements */
+	uint16_t	elmcnt;		/* Element Count (number used) */
+	uint16_t	elmsize;	/* Size of an individual raidCLine */
+	uint16_t	rchksum;	/* RAID table check sum
+					   (no rconfTblV2)*/
+	uint32_t	res1;		/* Reserved */
+	uint16_t	res2;		/* was bldRate - Time in 1/10s
+					   between idle build bursts */
+	uint16_t	res3;		/* was bldAmount - Block to build
+					   during a build burst */
+	uint32_t	raidFlags;
+	uint32_t	timestamp;	/* used for iROC. A stamp to find
+					   which is latest */
+	uint8_t		irocFlags;
+#define	ASR_IF_VERIFY_WITH_AUTOFIX	0x01
+#define ASR_IF_BOOTABLE			0x80
+	uint8_t		dirty;		/* Records "open state" for array */
+#define ARRAY_STATE_OK		0x00
+#define ARRAY_STATE_DIRTY	0x03
+	uint8_t		actionPriority;
+	uint8_t		spareid;	/* Stored in member disk meta data
+					   to declare the ID of dedicated
+					   spare to show up. */
+	uint32_t	sparedrivemagic;/* drivemagic (in RB) of the spare
+					   at above ID. */
+	uint32_t	raidmagic;	/* used to identify spare drive with
+					   its mirror set. */
+	uint32_t 	verifyDate;	/* used by iomgr */
+	uint32_t	recreateDate;	/* used by iomgr */
+	uint8_t		res4[12];	/* Reserved */
+	struct asr_raid_configline ent[RCTBL_MAX_ENTRIES];
+} __attribute__ ((packed));
+
+
+#define RBLOCK_VER  8           /* Version of the reserved block */
+#define B0RESRVD    0x37FC4D1E  /* Signature of the reserved block */
+#define SVALID      0x4450544D  /* ASCII code for "DPTM" DPT Mirror */
+
+struct asr_reservedblock
+{
+	uint32_t	b0idcode;	/* 0x00 - ID code signifying block 0
+					   reserved */
+	uint8_t		lunsave[8];	/* 0x04 - NOT USED - LUN mappings for
+					   all drives */
+	uint16_t	sdtype;		/* 0x0C - NOT USED - drive type in
+					   boot prom */
+	uint16_t	ssavecyl;	/* 0x0E - NOT USED - Set Parameters
+					   cylinders */
+	uint8_t		ssavehed;	/* 0x10 - NOT USED - Set Parameters
+					   heads */
+	uint8_t		ssavesec;	/* 0x11 - NOT USED - Set Parameters
+					   sectors */
+	uint8_t		sb0flags;	/* 0x12 - flags saved in reserved
+					   block */
+	uint8_t		jbodEnable;	/* 0x13 - jbod enable -- DEC drive
+					   hiding */
+	uint8_t		lundsave;	/* 0x14 - NOT USED - LUNMAP disable
+					   flags */
+	uint8_t		svpdirty;	/* 0x15 - NOT USED - saved percentage
+					   dirty */
+	uint16_t	biosInfo;	/* 0x16 - bios info - set by
+					   I2O_EXEC_BIOS_INFO_SET */
+	uint16_t	svwbskip;	/* 0x18 - NOT USED - saved write-back
+					   skip value */
+	uint16_t	svwbcln;	/* 0x1A - NOT USED - saved maximum
+					   clean blocks in write-back */
+	uint16_t	svwbmax;	/* 0x1C - NOT USED - saved maximum
+					   write-back length */
+	uint16_t	res3;		/* 0x1E - unused (was write-back burst
+					   block count) */
+	uint16_t	svwbmin;	/* 0x20 - NOT USED - saved minimum
+					   block count to write */
+	uint16_t	res4;		/* 0x22 - unused (was minimum
+					   look-ahead length) */
+	uint16_t	svrcacth;	/* 0x24 - NOT USED - saved read cache
+					   threshold */
+	uint16_t	svwcacth;	/* 0x26 - NOT USED - saved write
+					    cache threshold */
+	uint16_t	svwbdly;	/* 0x28 - NOT USED - saved write-back
+					   delay */
+	uint8_t		svsdtime;	/* 0x2A - NOT USED - saved spin down
+					   time */
+	uint8_t		res5;		/* 0x2B - unused */
+	uint16_t	firmval;	/* 0x2C - NOT USED - firmware on
+					   drive  (dw) */
+	uint16_t	firmbln;	/* 0x2E - NOT USED - length in blocks
+					   for firmware */
+	uint32_t	firmblk;	/* 0x30 - NOT USED - starting block
+					   for firmware */
+	uint32_t	fstrsvrb;	/* 0x34 - 1st block reserved by
+					   Storage Manager */
+	uint16_t	svBlockStorageTid;	/* 0x38 - */
+	uint16_t	svtid;		/* 0x3A - */
+	uint8_t		svseccfl;	/* 0x3C - NOT USED - reserved block
+					   scsi bus ecc flags */
+	uint8_t		res6;		/* 0x3D - unused */
+	uint8_t		svhbanum;	/* 0x3E - NOT USED - HBA's unique
+					   RAID number */
+	uint8_t		resver;		/* 0x3F - reserved block version
+					   number */
+	uint32_t	drivemagic;	/* 0x40 - Magic number of this drive -
+					   used w/ RCTBLs */
+	uint8_t		reserved[20];	/* 0x44 - unused */
+	uint8_t		testnum;	/* 0x58 - NOT USED - diagnostic test
+					   number */
+	uint8_t		testflags;	/* 0x59 - NOT USED - diagnostic test
+					   flags */
+	uint16_t	maxErrorCount;	/* 0x5A - NOT USED - diagnostic test
+					   maximum error count */
+	uint32_t	count;		/* 0x5C - NOT USED - diagnostic test
+					   cycles - # of iterations */
+	uint32_t	startTime;	/* 0x60 - NOT USED - diagnostic test
+					   absolute test start time in
+					   seconds */
+	uint32_t	interval;	/* 0x64 - NOT USED - diagnostic test
+					   interval in seconds */
+	uint8_t		tstxt0;		/* 0x68 - not used - originally
+					   diagnostic test exclusion period
+					   start hour */
+	uint8_t		tstxt1;		/* 0x69 - not used - originally
+					   diagnostic test exclusion period
+					   end hour */
+	uint8_t		serNum[32];	/* 0x6A - reserved */
+	uint8_t		res8[102];	/* 0x8A - reserved */
+	uint32_t	fwTestMagic;	/* 0xF0 - test magic number - used by
+					   FW Test for automated tests */
+	uint32_t	fwTestSeqNum;	/* 0xF4 - test sequence number - used
+					   by FW Test for automated tests */
+	uint8_t		fwTestRes[8];	/* 0xF6 - reserved by FW Test for
+					   automated tests */
+	uint32_t	smagic;		/* 0x100 - magic value saying software
+					   half is valid */
+	uint32_t	raidtbl;	/* 0x104 - pointer to first block of
+					   raid table */
+	uint16_t	raidline;	/* 0x108 - line number of this raid
+					   table entry - only if version <7 */
+	uint8_t		res9[0xF6];	/* 0x10A - reserved for software stuff*/
+} __attribute__ ((packed));
+
+
+
+#define ARRAY_NEW       0x00
+#define ARRAY_EQUAL     0x01
+#define ARRAY_SEQ_LESS  0x02
+#define ARRAY_SEQ_GREAT 0x03
+
+#define LOCK_PRIORITY      10	/* Uses 10, 11, 12 - For all three channels */
+
+
+/* B0FLAGS flag bits: */
+
+#define SMARTENA		7	/* SMART emulation enabled */
+#define CLRERROR		4	/* Clear stage of Interpret Format
+					   not completed */
+#define FMTERROR		3	/* Format stage of Interpret Format
+					   not completed */
+#define WRTTHRU			2	/* write throughs */
+#define CACHEDIS		0	/* cache disable bit */
+#define PREDICTIVE_ENABLE	0x02
+
+#define ID_MAP_PHYSICAL_M	1	/* Logical Map Physical */
+#define ID_MAP_LOGICAL_M	2	/* Either Dual Level or Single Level
+					   Logical*/
+
+#define MAX_LSU_COUNT			256
+
+#define MAX_LSU_COMPONENTS		64
+#define MAX_LSU_INQUIRY_DATA		64
+#define MAX_LSU_SERIAL_NUMBER		8
+#define STD_INQUIRY_SIZE		48
+
+#define MAX_LSU_BAD_BLOCKS		8
+
+/* lsuType definitions */
+#define LT_UNCONFIGURED_M		0x00
+#define LT_RAID0_M			0x01
+#define LT_RAID1_M			0x02
+#define LT_RAID3_M			0x08
+#define LT_RAID4_M			0x10
+#define LT_RAID5_M			0x20
+#define LT_REDIR_M			0x40
+#define LT_SPARE_M			0x80
+
+#define	LSU_RAID_LEVEL_0		LT_RAID0_M
+#define	LSU_RAID_LEVEL_1		LT_RAID1_M
+#define	LSU_RAID_LEVEL_3		LT_RAID3_M
+#define	LSU_RAID_LEVEL_4		LT_RAID4_M
+#define	LSU_RAID_LEVEL_5		LT_RAID5_M
+#define LSU_RAID_REDIRECT		LT_REDIR_M
+#define	LSU_RAID_SPARE			LT_SPARE_M
+
+/* raidState definitions */
+#define LS_OPTIMAL_M			0x00
+#define LS_DEGRADED_M			0x01
+#define LS_REBUILDING_M			0x02
+#define LS_MORPHING_M			0x03
+#define LS_DEAD_M			0x04
+#define LS_WARNING_M			0x05
+
+#define LS_VERIFYING_M			0x0A
+#define LSU_ARRAY_SUBSTATE_WITH_FIX	0x10
+
+#define LS_BUILDING_M			0x0B
+#define LS_CREATED_M			0x54
+#define LS_DIAGNOSING_M			0xFD
+
+
+/* arrayState definitions */
+#define LSU_ARRAY_STATE_OPTIMAL			0x00
+#define LSU_ARRAY_STATE_DEGRADED		0x01
+/* etc. */
+
+/* component raidState definitions */
+#define MAIN_STATE				0x0F
+#define LSU_COMPONENT_STATE_OPTIMAL		0x00
+#define LSU_COMPONENT_STATE_DEGRADED		0x01
+#define LSU_COMPONENT_STATE_UNCONFIGURED	0x02
+#define LSU_COMPONENT_STATE_FAILED		0x03
+#define LSU_COMPONENT_STATE_REPLACED		0x04
+#define LSU_COMPONENT_STATE_UNINITIALIZED	0x0A
+
+#define LSU_COMPONENT_SUBSTATE_BUILDING		0x10  /* drive is being built
+							 for first time */
+#define LSU_COMPONENT_SUBSTATE_REBUILDING	0x20  /* drive is being
+							 rebuilt */
+#define LSU_ARRAY_SUBSTATE_AWAIT_FORMAT		0x50
+/* etc. */
+
+/* Beginning of other stuff that Darrick Wong added */
+
+struct asr {
+	struct asr_reservedblock rb;
+	struct asr_raidtable *rt;
+};
+
+#endif /* FORMAT_HANDLER */
+
+int register_asr(struct lib_context *lc);
+
+#endif /* _ASR_H */
/cvs/dm/dmraid/lib/format/ataraid/hpt37x.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/hpt37x.c
+++ -	2008-02-22 16:57:41.082281000 +0000
@@ -0,0 +1,413 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Highpoint 37X ATARAID metadata format handler.
+ *
+ * hpt37x_read(), hpt37x_group() and group_rd() profited from
+ * Carl-Daniel Hailfinger's raiddetect code.
+ */
+#define	HANDLER "hpt37x"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "hpt37x.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/* Make up RAID set name from magic_[01] numbers */
+/* FIXME: better name ? */
+static size_t _name(struct hpt37x *hpt, char *str, size_t len,
+		    unsigned int subset)
+{
+	const char *fmt;
+
+	if (hpt->magic_0)
+		fmt = (subset &&
+		       (hpt->type == HPT37X_T_RAID01_RAID0 ||
+		        hpt->type == HPT37X_T_RAID01_RAID1)) ?
+		       "hpt37x_%u-%u" : "hpt37x_%u";
+	else
+		fmt = "hpt37x_SPARE";
+
+	/* FIXME: hpt->order not zero-based. */
+	return snprintf(str, len, fmt,
+			hpt->magic_1 ? hpt->magic_1 : hpt->magic_0, hpt->order);
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret;
+	struct hpt37x *hpt = META(rd, hpt37x);
+
+	if ((ret = dbg_malloc((len = _name(hpt, NULL, 0, subset) + 1)))) {
+		_name(hpt, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN -
+			 (strrchr(ret, '-') ? 3 : 1));
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/*
+ * Retrieve status of device.
+ * FIXME: is this sufficient to cover all state ?
+ */
+static enum status status(struct hpt37x *hpt)
+{
+	return hpt->magic == HPT37X_MAGIC_BAD ? s_broken : s_ok;
+}
+
+/* Mapping of HPT 37X types to generic types. */
+static struct types types[] = {
+	{ HPT37X_T_SINGLEDISK, t_linear},
+	{ HPT37X_T_SPAN, t_linear},
+	{ HPT37X_T_RAID0, t_raid0},
+	{ HPT37X_T_RAID1, t_raid1},
+	{ HPT37X_T_RAID01_RAID0, t_raid0},
+	{ HPT37X_T_RAID01_RAID1, t_raid1},
+	/* FIXME: support RAID 3+5 */
+	{ 0, t_undef}
+};
+
+/* Neutralize disk type. */
+static enum type type(struct hpt37x *hpt)
+{
+	return hpt->magic_0 ?
+	       rd_type(types, (unsigned int) hpt->type) : t_spare;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD(new), hpt37x))->disk_number <
+	       (META(RD(pos), hpt37x))->disk_number;
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD_RS(RS(new)), hpt37x))->order <
+	       (META(RD_RS(RS(pos)), hpt37x))->order;
+}
+
+/*
+ * Read a Highpoint 37X RAID device.
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct hpt37x *hpt = meta;
+	struct hpt37x_errorlog *l;
+
+	CVT32(hpt->magic);
+	CVT32(hpt->magic_0);
+	CVT32(hpt->magic_1);
+	CVT32(hpt->order);
+	CVT32(hpt->total_secs);
+	CVT32(hpt->disk_mode);
+	CVT32(hpt->boot_mode);
+
+	for (l = hpt->errorlog;
+	     l < hpt->errorlog + hpt->error_log_entries;
+	     l++) {
+		CVT32(l->timestamp);
+		CVT32(l->lba);
+	}
+}
+#endif
+
+/* Magic check. */
+static int is_hpt37x(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct hpt37x *hpt = meta;
+
+	return (hpt->magic == HPT37X_MAGIC_OK ||
+		hpt->magic == HPT37X_MAGIC_BAD) &&
+		hpt->disk_number < 8;
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *hpt37x_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct hpt37x), HPT37X_CONFIGOFFSET,
+			     to_cpu, is_hpt37x, NULL, setup_rd, handler);
+}
+
+/*
+ * Write a Highpoint 37X RAID device.
+ */
+static int hpt37x_write(struct lib_context *lc,
+			struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct hpt37x *hpt = META(rd, hpt37x);
+
+	to_disk(hpt);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(hpt);
+#endif
+
+	return ret;
+}
+
+/*
+ * Group the RAID disk into a set.
+ *
+ * Check device hierarchy and create sub sets appropriately.
+ *
+ */
+static unsigned int stride(struct hpt37x *hpt)
+{
+	return hpt->raid0_shift ? 1 << hpt->raid0_shift : 0;
+}
+
+static int mismatch(struct lib_context *lc, struct raid_dev *rd, char magic)
+{
+	LOG_ERR(lc, 0, "%s: magic_%c mismatch on %s",
+		handler, magic, rd->di->path);
+}
+
+static void super_created(struct raid_set *ss, void *private)
+{
+	struct hpt37x *hpt = META(private, hpt37x);
+
+	ss->type   = hpt->type == HPT37X_T_RAID01_RAID0 ? t_raid1 : t_raid0;
+	ss->stride = stride(hpt);
+}
+
+/* FIXME: handle spares in mirrors and check that types are correct. */
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct hpt37x *h, *hpt = META(rd, hpt37x);
+
+	if (!init_raid_set(lc, rs, rd, stride(hpt), hpt->type, handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+	h = DEVS(rs) ? META(RD_RS(rs), hpt37x) : NULL;
+
+	switch (hpt->type) {
+	case HPT37X_T_SINGLEDISK:
+	case HPT37X_T_SPAN:
+	case HPT37X_T_RAID0:
+	case HPT37X_T_RAID1:
+		if (h && h->magic_0 != hpt->magic_0)
+			return mismatch(lc, rd, '0');
+
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case HPT37X_T_RAID01_RAID0:
+	case HPT37X_T_RAID01_RAID1:
+		if (h && h->magic_1 != hpt->magic_1)
+			return mismatch(lc, rd, '1');
+
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+/*
+ * Add a Highpoint RAID device to a set.
+ */
+static struct raid_set *hpt37x_group(struct lib_context *lc,
+				       struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/*
+ * Check a Highpoint 37X RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	return (META(rd, hpt37x))->raid_disks;
+}
+
+static int check_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_dev *rd, void *context)
+{
+	/*
+	 * FIXME: raid_disks member wrong ?
+	 *	  (eg, Peter Jonas RAID1 metadata, 2 disks and raid_disks = 1)
+	 */
+	return T_RAID1(rd);
+}
+
+static int hpt37x_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL, check_rd, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct hpt37x *hpt = META(rd, hpt37x);
+
+	/* Avoid write trashing. */
+	if (status(hpt) & s_broken)
+		return 0;
+
+	hpt->magic = HPT37X_MAGIC_BAD;
+
+	return 1;
+}
+
+static struct event_handlers hpt37x_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about an HPT37X RAID device.
+ */
+static void hpt37x_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	struct hpt37x *hpt = META(rd, hpt37x);
+	struct hpt37x_errorlog *el;
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	DP("magic: 0x%x", hpt, hpt->magic);
+	DP("magic_0: 0x%x", hpt, hpt->magic_0);
+	DP("magic_1: 0x%x", hpt, hpt->magic_1);
+	DP("order: %u", hpt, hpt->order);
+	DP("raid_disks: %u", hpt, hpt->raid_disks);
+	DP("raid0_shift: %u", hpt, hpt->raid0_shift);
+	DP("type: %u", hpt, hpt->type);
+	DP("disk_number: %u", hpt, hpt->disk_number);
+	DP("total_secs: %u", hpt, hpt->total_secs);
+	DP("disk_mode: 0x%x", hpt, hpt->disk_mode);
+	DP("boot_mode: 0x%x", hpt, hpt->boot_mode);
+	DP("boot_disk: %u", hpt, hpt->boot_disk);
+	DP("boot_protect: %u", hpt, hpt->boot_protect);
+	DP("error_log_entries: %u", hpt, hpt->error_log_entries);
+	DP("error_log_index: %u", hpt, hpt->error_log_index);
+	if (hpt->error_log_entries)
+		log_print(lc, "error_log:");
+
+	for (el = hpt->errorlog; el < hpt->errorlog + 32; el++) {
+		if (!el->timestamp)
+			break;
+
+		DP("timestamp: %u", hpt, el->timestamp);
+		DP("reason: %u", hpt, el->reason);
+		DP("disk: %u", hpt, el->disk);
+		DP("status: %u", hpt, el->status);
+		DP("sectors: %u", hpt, el->sectors);
+		DP("lba: %u", hpt, el->lba);
+       };
+}
+#endif
+
+static struct dmraid_format hpt37x_format = {
+	.name	= HANDLER,
+	.descr	= "Highpoint HPT37X",
+	.caps	= "S,0,1,10,01",
+	.format = FMT_RAID,
+	.read	= hpt37x_read,
+	.write	= hpt37x_write,
+	.group	= hpt37x_group,
+	.check	= hpt37x_check,
+	.events	= &hpt37x_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= hpt37x_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_hpt37x(struct lib_context *lc)
+{
+	return register_format_handler(lc, &hpt37x_format);
+}
+
+/* Calculate RAID device size in sectors depending on RAID type. */
+static uint64_t sectors(struct raid_dev *rd, struct hpt37x *hpt)
+{
+	uint64_t ret = 0;
+	struct dev_info *di = rd->di;
+
+	switch (rd->type) {
+	case t_raid0:
+		ret = hpt->total_secs / (hpt->raid_disks ? hpt->raid_disks : 1);
+		break;
+
+	case t_raid1:
+		ret = hpt->total_secs;
+		break;
+
+	default:
+		ret = di->sectors;
+	}
+
+	/* Subtract offset sectors on drives > 0. */
+	return ret - rd->offset;
+}
+
+/* Derive the RAID device contents from the Highpoint ones. */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct hpt37x *hpt = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = HPT37X_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*hpt);
+	rd->meta_areas->area = (void*) hpt;
+
+	rd->di  = di;
+	rd->fmt = &hpt37x_format;
+
+	rd->status = status(hpt);
+	rd->type   = type(hpt);
+
+	/* Data offset from start of device; first device is special */
+	rd->offset  = hpt->disk_number ? HPT37X_DATAOFFSET : 0;
+        if (!(rd->sectors = sectors(rd, hpt)))
+		return log_zero_sectors(lc, di->path, handler);
+
+        return (rd->name = name(lc, rd, 1)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/hpt37x.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/hpt37x.h
+++ -	2008-02-22 16:57:41.165768000 +0000
@@ -0,0 +1,100 @@
+/*
+ * Copyright (c) 2000,2001 Søren Schmidt <sos at FreeBSD.org>
+ * All rights reserved.
+ *
+ * dmraid modifications:
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer,
+ *    without modification, immediately at the beginning of the file.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ * 3. The name of the author may not be used to endorse or promote products
+ *    derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR `AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef	_HPT37X_H_
+#define	_HPT37X_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+/* HPT 37x config data byte offset on disk */
+#define HPT37X_CONFIGOFFSET	(9 << 9) /* 9 sectors */
+#define HPT37X_DATAOFFSET	10	/* Data offset in sectors */
+
+/* Ondisk metadata for Highpoint ATARAID */
+struct hpt37x {
+       uint8_t  filler1[32];
+
+       uint32_t       magic;
+#define HPT37X_MAGIC_OK   0x5a7816f0
+#define HPT37X_MAGIC_BAD  0x5a7816fd  
+
+       uint32_t       magic_0;	/* Set identifier */
+       uint32_t       magic_1;	/* Array identifier */
+
+       uint32_t       order;  
+#define HPT_O_MIRROR   0x01  
+#define HPT_O_STRIPE   0x02
+#define HPT_O_OK       0x04
+
+       uint8_t        raid_disks;
+       uint8_t        raid0_shift; 
+
+       uint8_t        type;
+#define HPT37X_T_RAID0		0x00 
+#define HPT37X_T_RAID1		0x01
+#define HPT37X_T_RAID01_RAID0	0x02
+#define HPT37X_T_SPAN		0x03
+#define HPT37X_T_RAID_3		0x04   
+#define HPT37X_T_RAID_5		0x05
+#define HPT37X_T_SINGLEDISK	0x06
+#define HPT37X_T_RAID01_RAID1	0x07
+
+       uint8_t        disk_number;
+       uint32_t       total_secs; 
+       uint32_t       disk_mode;  
+       uint32_t       boot_mode;
+       uint8_t        boot_disk; 
+       uint8_t        boot_protect;
+       uint8_t        error_log_entries;
+       uint8_t        error_log_index;  
+       struct hpt37x_errorlog
+       {
+	       uint32_t       timestamp;
+	       uint8_t        reason;   
+#define HPT_R_REMOVED          0xfe      
+#define HPT_R_BROKEN           0xff      
+
+	       uint8_t        disk;
+	       uint8_t        status;
+	       uint8_t        sectors;
+	       uint32_t       lba;
+       } errorlog[32];
+       uint8_t        filler[60];
+} __attribute__ ((packed));
+#endif
+
+int register_hpt37x(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/hpt45x.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/hpt45x.c
+++ -	2008-02-22 16:57:41.269910000 +0000
@@ -0,0 +1,359 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Highpoint 45X ATARAID metadata format handler.
+ *
+ * hpt45x_read(), hpt45x_group() and group_rd() profited from
+ * Carl-Daniel Hailfinger's raiddetect code.
+ */
+#define	HANDLER "hpt45x"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "hpt45x.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/* Make up RAID set name from magic_0 number */
+/* FIXME: better name ? */
+static size_t _name(struct hpt45x *hpt, char *str, size_t len,
+		    unsigned int subset)
+{
+	const char *fmt;
+
+	if (hpt->magic_0)
+		fmt = subset ? "hpt45x_%u-%u" : "hpt45x_%u";
+	else
+		fmt = "hpt45x_SPARE";
+
+	return snprintf(str, len, fmt, hpt->magic_0, hpt->raid1_disk_number);
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret;
+	struct hpt45x *hpt = META(rd, hpt45x);
+
+	if ((ret = dbg_malloc((len = _name(hpt, NULL, 0, subset) + 1)))) {
+		_name(hpt, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN -
+			 (strrchr(ret, '-') ? 3 : 1));
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/*
+ * Retrieve status of device.
+ * FIXME: is this sufficient to cover all state ?
+ */
+static enum status status(struct hpt45x *hpt)
+{
+	return hpt->magic == HPT45X_MAGIC_BAD ? s_broken : s_ok;
+}
+
+/* Mapping of HPT 45X types to generic types */
+static struct types types[] = {
+	{ HPT45X_T_SPAN, t_linear},
+	{ HPT45X_T_RAID0, t_raid0},
+	{ HPT45X_T_RAID1, t_raid1},
+/* FIXME: handle RAID 3+5 */
+	{ 0, t_undef}
+};
+
+/* Neutralize disk type */
+static enum type type(struct hpt45x *hpt)
+{
+	return hpt->magic_0 ? rd_type(types, (unsigned int) hpt->type) :
+	       t_spare;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD(new), hpt45x))->disk_number <
+	       (META(RD(pos), hpt45x))->disk_number;
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD_RS(RS(new)), hpt45x))->raid1_disk_number <
+	       (META(RD_RS(RS(pos)), hpt45x))->raid1_disk_number;
+}
+
+/*
+ * Group the RAID disk into a HPT45X set.
+ *
+ * Check device hierarchy and create super set appropriately.
+ */
+static unsigned int stride(unsigned int shift)
+{
+	return shift ? 1 << shift : 0;
+}
+
+static void super_created(struct raid_set *super, void *private)
+{
+	super->type   = t_raid1;
+	super->stride = stride(META((private), hpt45x)->raid1_shift);
+}
+
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct hpt45x *hpt = META(rd, hpt45x);
+
+	if (!init_raid_set(lc, rs, rd, stride(hpt->raid0_shift),
+			   hpt->type, handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (hpt->type) {
+	case HPT45X_T_SPAN:
+	case HPT45X_T_RAID1:
+  no_raid10:
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case HPT45X_T_RAID0:
+		if (hpt->raid1_type != HPT45X_T_RAID1)
+			goto no_raid10;
+		/*
+		 * RAID10:
+		 *
+		 * We've got a striped raid set with a mirror on top
+		 * when we get here.
+		 * Let's find and optionally allocate the mirror set on top.
+		 */
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+/*
+ * Add a Highpoint RAID device to a set.
+ */
+static struct raid_set *hpt45x_group(struct lib_context *lc,
+				       struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/*
+ * Read a Highpoint 45X RAID device.
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct hpt45x *hpt = meta;
+
+	CVT32(hpt->magic);
+	CVT32(hpt->magic_0);
+	CVT32(hpt->magic_1);
+	CVT32(hpt->total_secs);
+}
+#endif
+
+/* Magic check. */
+static int is_hpt45x(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct hpt45x *hpt = meta;
+
+	return (hpt->magic == HPT45X_MAGIC_OK ||
+		hpt->magic == HPT45X_MAGIC_BAD) &&
+		hpt->disk_number < 8;
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *hpt45x_read(struct lib_context *lc,
+				      struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct hpt45x), HPT45X_CONFIGOFFSET,
+			     to_cpu, is_hpt45x, NULL, setup_rd, handler);
+}
+
+/*
+ * Write a Highpoint 45X RAID device.
+ */
+static int hpt45x_write(struct lib_context *lc,
+			struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct hpt45x *hpt = META(rd, hpt45x);
+
+	to_disk(hpt);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(hpt);
+#endif
+	return ret;
+}
+
+/*
+ * Check a Highpoint 45X RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	return (META(rd, hpt45x))->raid_disks;
+}
+
+static int hpt45x_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL,
+			      NO_CHECK_RD, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct hpt45x *hpt = META(rd, hpt45x);
+
+	/* Avoid write trashing. */
+	if (S_BROKEN(status(hpt)))
+		return 0;
+
+	hpt->magic = HPT45X_MAGIC_BAD;
+
+	return 1;
+}
+
+static struct event_handlers hpt45x_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about an HPT45X RAID device.
+ */
+static void hpt45x_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int i;
+	struct hpt45x *hpt = META(rd, hpt45x);
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	DP("magic: 0x%x", hpt, hpt->magic);
+	DP("magic_0: 0x%x", hpt, hpt->magic_0);
+	DP("magic_1: 0x%x", hpt, hpt->magic_1);
+	DP("total_secs: %u", hpt, hpt->total_secs);
+	DP("type: %u", hpt, hpt->type);
+	DP("raid_disks: %u", hpt, hpt->raid_disks);
+	DP("disk_number: %u", hpt, hpt->disk_number);
+	DP("raid0_shift: %u", hpt, hpt->raid0_shift);
+	for (i = 0; i < 3; i++)
+		P2("dummy[%u]: 0x%x", hpt, i, hpt->dummy[i]);
+	DP("raid1_type: %u", hpt, hpt->raid1_type);
+	DP("raid1_raid_disks: %u", hpt, hpt->raid1_raid_disks);
+	DP("raid1_disk_number: %u", hpt, hpt->raid1_disk_number);
+	DP("raid1_shift: %u", hpt, hpt->raid1_shift);
+
+	for (i = 0; i < 3; i++)
+		P2("dummy1[%u]: 0x%x", hpt, i, hpt->dummy1[i]);
+}
+#endif
+
+static struct dmraid_format hpt45x_format = {
+	.name	= HANDLER,
+	.descr	= "Highpoint HPT45X",
+	.caps	= "S,0,1,10",
+	.format = FMT_RAID,
+	.read	= hpt45x_read,
+	.write	= hpt45x_write,
+	.group	= hpt45x_group,
+	.check	= hpt45x_check,
+	.events	= &hpt45x_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= hpt45x_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_hpt45x(struct lib_context *lc)
+{
+	return register_format_handler(lc, &hpt45x_format);
+}
+
+/* Calculate RAID device size in sectors depending on RAID type. */
+static uint64_t sectors(struct raid_dev *rd, void *meta)
+{
+	struct hpt45x *hpt = meta;
+
+	switch (rd->type) {
+	case t_raid0:
+		return hpt->total_secs /
+		       (hpt->raid_disks ? hpt->raid_disks : 1);
+
+	case t_raid1:
+		return hpt->total_secs;
+
+	default:
+		return rd->meta_areas->offset;
+	}
+}
+
+/* Set the RAID device contents up derived from the Highpoint ones */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct hpt45x *hpt = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = HPT45X_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*hpt);
+	rd->meta_areas->area = (void*) hpt;
+
+	rd->di = di;
+	rd->fmt = &hpt45x_format;
+
+	rd->status = status(hpt);
+	rd->type   = type(hpt);
+
+	rd->offset = HPT45X_DATAOFFSET;
+	if (!(rd->sectors = sectors(rd, hpt)))
+		return log_zero_sectors(lc, di->path, handler);
+
+	return (rd->name = name(lc, rd, hpt->raid1_type == HPT45X_T_RAID1)) ?
+	       1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/hpt45x.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/hpt45x.h
+++ -	2008-02-22 16:57:41.351114000 +0000
@@ -0,0 +1,53 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_HPT45X_H_
+#define	_HPT45X_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+/* Highpoint 45x config data sector offset off end of disk */
+#define HPT45X_CONFIGOFFSET	((di->sectors - 11) << 9)
+#define HPT45X_DATAOFFSET	0 /* Data offset in sectors */
+
+/* Ondisk metadata for Highpoint 45X ATARAID */
+struct hpt45x {
+	uint32_t	magic;		/* 0x0 - 0x03 */
+#define HPT45X_MAGIC_OK   0x5a7816f3
+#define HPT45X_MAGIC_BAD  0x5a7816fd
+
+	uint32_t	magic_0;	/* 0x04 - 0x07 Set identifier */
+	uint32_t	magic_1;	/* 0x08 - 0x0A (Sub-)Array identifier */
+
+	uint32_t	total_secs; 	/* 0x0B - 0x0F */
+
+	uint8_t	type;		/* 0x10 */
+#define HPT45X_T_SPAN	0x04
+#define HPT45X_T_RAID0	0x05
+#define HPT45X_T_RAID1	0x06
+
+	uint8_t	raid_disks;	/* 0x11 */
+	uint8_t	disk_number;	/* 0x12 */
+	uint8_t	raid0_shift; 	/* 0x13 */
+
+	uint32_t	dummy[3];	/* 0x14 - 0x1F */
+
+	uint8_t	raid1_type;		/* 0x20 */
+	uint8_t	raid1_raid_disks;	/* 0x21 */
+	uint8_t	raid1_disk_number;	/* 0x22 */
+	uint8_t	raid1_shift;		/* 0x23 */
+
+	uint32_t	dummy1[3];	/* 0x24 - 0x2F */
+} __attribute__ ((packed));
+#endif
+
+int register_hpt45x(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/isw.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/isw.c
+++ -	2008-02-22 16:57:41.430673000 +0000
@@ -0,0 +1,801 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Intel Software RAID metadata format handler.
+ *
+ * isw_read() etc. profited from Carl-Daniel Hailfinger's raiddetect code.
+ *
+ * Profited from the Linux 2.4 iswraid driver by
+ * Boji Tony Kannanthanam and Martins Krikis.
+ */
+#define	HANDLER	"isw"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "isw.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/*
+ * Make up RAID set name from family_num and volume name.
+ */
+static size_t _name(struct isw *isw, struct isw_dev *dev,
+		     char *str, size_t len)
+{
+	return snprintf(str, len, dev ? "isw_%u_%s" : "isw_%u",
+			isw->family_num, (char*) dev->volume);
+}
+
+static char *name(struct lib_context *lc, struct isw *isw, struct isw_dev *dev)
+{
+        size_t len;
+        char *ret;
+
+        if ((ret = dbg_malloc((len = _name(isw, dev, NULL, 0) + 1)))) {
+                _name(isw, dev, ret, len);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN -
+			 (dev ? strlen((char*) dev->volume) - 2 : 1));
+        } else
+		log_alloc_err(lc, handler);
+
+        return ret;
+}
+
+/* Find a disk table slot by serial number. */
+static struct isw_disk *_get_disk(struct isw *isw, struct dev_info *di)
+{
+	if (di->serial) {
+		struct isw_disk *disk = isw->disk;
+
+		do {
+			if (!strncmp(di->serial, (const char*) disk->serial,
+				     MAX_RAID_SERIAL_LEN))
+				return disk;
+		} while (++disk < isw->disk + isw->num_disks);
+	}
+
+	return NULL;
+}
+
+static struct isw_disk *get_disk(struct lib_context *lc,
+				 struct dev_info *di, struct isw *isw)
+{
+	struct isw_disk *disk;
+
+	if ((disk = _get_disk(isw, di)))
+		return disk;
+
+	LOG_ERR(lc, NULL, "%s: Error finding disk table slot for %s",
+		handler, di->path);
+}
+
+/*
+ * Retrieve status of device.
+ *
+ * FIXME: is this sufficient to cover all state ?
+ */
+static enum status __status(unsigned int status)
+{
+	return ((status & (CONFIGURED_DISK|USABLE_DISK)) &&
+		!(FAILED_DISK & status)) ?
+	       s_ok : s_broken;
+}
+
+static enum status status(struct lib_context *lc, struct raid_dev *rd)
+{
+	struct isw_disk *disk;
+
+	if ((disk = get_disk(lc, rd->di, META(rd, isw))))
+		return __status(disk->status);
+
+	return s_undef;
+}
+
+/*
+ * Mapping of Intel types to generic types.
+ */
+static struct types types[] = {
+        { ISW_T_RAID0, t_raid0},
+        { ISW_T_RAID1, t_raid1},
+        { ISW_T_RAID5, t_raid5_la},
+        { 0, t_undef},
+};
+
+/* Neutralize disk type. */
+static enum type type(struct raid_dev *rd)
+{
+	struct isw_dev *dev = rd->private.ptr;
+
+	return dev ? rd_type(types, (unsigned int) dev->vol.map.raid_level) :
+		     t_group;
+}
+
+/*
+ * Generate checksum of Raid metadata for mpb_size/sizeof(u32) words
+ * (checksum field itself ignored for this calculation).
+ */
+static uint32_t _checksum(struct isw *isw)
+{
+	uint32_t end = isw->mpb_size / sizeof(end),
+		 *p = (uint32_t*) isw, ret = 0;
+
+	while (end--)
+		ret += *p++;
+
+	return ret - isw->check_sum;
+}
+
+/* Calculate next isw device offset. */
+static struct isw_dev *advance_dev(struct isw_dev *dev,
+				   struct isw_map *map, size_t add)
+{
+	return (struct isw_dev*) ((uint8_t*) dev +
+				  (map->num_members - 1) * 
+				  sizeof(map->disk_ord_tbl) + add);
+}
+
+/* Advance to the next isw_dev from a given one. */
+static struct isw_dev *advance_raiddev(struct isw_dev *dev)
+{
+	struct isw_vol *vol = &dev->vol;
+	struct isw_map *map = &vol->map;
+
+	/* Correction: yes, it sits here! */
+	dev = advance_dev(dev, map, sizeof(*dev));
+
+	if (vol->migr_state) /* need to add space for another map */
+		dev = advance_dev(dev, map, sizeof(*map));
+
+	return dev;
+}
+
+/* Return isw_dev by table index. */
+static struct isw_dev *raiddev(struct isw *isw, unsigned int i)
+{
+	struct isw_dev *dev = (struct isw_dev*) (isw->disk + isw->num_disks);
+
+	while (i--)
+		dev = advance_raiddev(dev);
+
+	return dev;
+}
+
+/*
+ * Read an Intel RAID device
+ */
+/* Endianess conversion. */
+enum convert { FULL, FIRST, LAST };
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu(x, y)
+#else
+/*
+ * We can differ from the read_raid_dev template here,
+ * because we don't get called from there.
+ */
+static void to_cpu(struct isw *isw, enum convert cvt)
+{
+	unsigned int i, j;
+	struct isw_disk *dsk;
+	struct isw_dev *dev;
+
+	if (cvt == FIRST || cvt == FULL) {
+		CVT32(isw->check_sum);
+		CVT32(isw->mpb_size);
+		CVT32(isw->family_num);
+		CVT32(isw->generation_num);
+	}
+
+	if (cvt == FIRST)
+		return;
+
+	for (dsk = isw->disk; dsk < &isw->disk[isw->num_disks]; dsk++) {
+		CVT32(dsk->totalBlocks);
+		CVT32(dsk->scsiId);
+		CVT32(dsk->status);
+	}
+
+	for (i = 0; i < isw->num_raid_devs; i++) {
+		dev = raiddev(isw, i);
+
+		/* RAID device. */
+		CVT32(dev->SizeLow);
+		CVT32(dev->SizeHigh);
+		CVT32(dev->status);
+		CVT32(dev->reserved_blocks);
+
+		/* RAID volume has 8 bit members only. */
+	
+		/* RAID map. */
+		CVT32(dev->vol.map.pba_of_lba0);
+		CVT32(dev->vol.map.blocks_per_member);
+		CVT32(dev->vol.map.num_data_stripes);
+		CVT16(dev->vol.map.blocks_per_strip);
+
+		for (j = 0; j < dev->vol.map.num_members; j++)
+			CVT16(dev->vol.map.disk_ord_tbl[j]);
+	}
+}
+#endif
+
+static int is_isw(struct lib_context *lc, struct dev_info *di, struct isw *isw)
+{
+	if (strncmp((const char *) isw->sig, MPB_SIGNATURE, sizeof(MPB_SIGNATURE) - 1))
+		return 0;
+
+	/* Check version info, older versions supported */
+	if (strncmp((const char*) isw->sig + sizeof(MPB_SIGNATURE) - 1,
+		    MPB_VERSION_RAID2, sizeof(MPB_VERSION_RAID2) - 1) > 0)
+		log_print(lc, "%s: untested metadata version %s found on %s",
+			  handler, isw->sig + sizeof(MPB_SIGNATURE) - 1,
+			  di->path);
+
+	return 1;
+}
+
+static void isw_file_metadata(struct lib_context *lc, struct dev_info *di,
+			      void *meta)
+{
+	struct isw *isw = meta;
+
+	/* Get the rounded up value for the metadata size */
+	size_t size = round_up(isw->mpb_size, ISW_DISK_BLOCK_SIZE);
+
+	file_metadata(lc, handler, di->path,
+		      meta + (size / ISW_DISK_BLOCK_SIZE > 1 ?
+			      ISW_DISK_BLOCK_SIZE : 0),
+		      size, (di->sectors - (size / ISW_DISK_BLOCK_SIZE)) << 9);
+	file_dev_size(lc, handler, di);
+}
+
+static int isw_read_extended(struct lib_context *lc, struct dev_info *di,
+			     struct isw **isw,
+			     uint64_t *isw_sboffset, size_t *size)
+{
+	struct isw *isw_tmp;
+
+	/* Get the rounded up value for the metadata blocks */
+	size_t blocks = div_up((*isw)->mpb_size, ISW_DISK_BLOCK_SIZE);
+
+	/* No extended metadata to read ? */
+	if (blocks < 2)
+		return 1;
+
+	/*
+	 * Allocate memory for the extended Intel superblock
+	 * and read it in. Reserve one more disk block in order
+	 * to be able to file the metadata in the proper sequence.
+	 * (ie, sectors 1, 2-n, 1 in core so that the filing can start at 2).
+	 */
+	*size = blocks * ISW_DISK_BLOCK_SIZE;
+	*isw_sboffset -= *size - ISW_DISK_BLOCK_SIZE;
+
+	if ((isw_tmp = alloc_private(lc, handler,
+				     *size + ISW_DISK_BLOCK_SIZE))) {
+		/* Read extended metadata to offset ISW_DISK_BLOCK_SIZE */
+		if (read_file(lc, handler, di->path,
+			      (void*) isw_tmp + ISW_DISK_BLOCK_SIZE,
+			      *size - ISW_DISK_BLOCK_SIZE, *isw_sboffset))
+			/* Copy in first metadata sector. */
+			memcpy(isw_tmp, *isw, ISW_DISK_BLOCK_SIZE);
+		else {
+			dbg_free(isw_tmp);
+			isw_tmp = NULL;
+		}
+	}
+
+	dbg_free(*isw);
+	*isw = isw_tmp;
+
+	return isw_tmp ? 1 : 0;
+}
+
+/* Check for RAID disk ok. */
+static int disk_ok(struct lib_context *lc, struct dev_info *di, struct isw *isw)
+{
+	struct isw_disk *disk = get_disk(lc, di, isw);
+
+	return disk && __status(disk->status) == s_ok;
+}
+
+static void *isw_read_metadata(struct lib_context *lc, struct dev_info *di,
+			       size_t *sz, uint64_t *offset,
+			       union read_info *info)
+{
+	size_t size = ISW_DISK_BLOCK_SIZE;
+	uint64_t isw_sboffset = ISW_CONFIGOFFSET;
+	struct isw *isw;
+
+	if (!(isw = alloc_private_and_read(lc, handler, size,
+					   di->path, isw_sboffset)))
+		goto out;
+
+	/*
+	 * Convert start of metadata only, because we might need to
+	 * read extended metadata located ahead of it first.
+	 */
+	to_cpu(isw, FIRST);
+
+	/* Check Signature and read optional extended metadata. */
+	if (!is_isw(lc, di, isw) ||
+	    !isw_read_extended(lc, di, &isw, &isw_sboffset, &size))
+		goto bad;
+
+	/*
+	 * Now that we made sure, that we've got all the
+	 * metadata, we can convert it completely.
+	 */
+	to_cpu(isw, LAST);
+
+	if (disk_ok(lc, di, isw)) {
+		*sz = size;
+		*offset = isw_sboffset;
+		info->u64 = isw_sboffset;
+		goto out;
+	}
+	
+   bad:
+	dbg_free(isw);
+	isw = NULL;
+
+   out:
+	return (void*) isw;
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *isw_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, isw_read_metadata, 0, 0, NULL, NULL,
+			     isw_file_metadata, setup_rd, handler);
+}
+
+/*
+ * Write an Intel Software RAID device.
+ */
+static int isw_write(struct lib_context *lc, struct raid_dev *rd, int erase)
+{
+	int ret;
+	struct isw *isw = META(rd, isw);
+
+	to_disk(isw, FULL);
+
+	/*
+	 * Copy 1st metadata sector to after the extended ones
+	 * and increment metadata area pointer by one block, so
+	 * that the metadata is filed in the proper sequence.
+	 */
+	memcpy((void*) isw + rd->meta_areas->size, isw, ISW_DISK_BLOCK_SIZE);
+	rd->meta_areas->area += ISW_DISK_BLOCK_SIZE;
+
+	ret = write_metadata(lc, handler, rd, -1, erase);
+
+	/* Correct metadata area pointer. */
+	rd->meta_areas->area -= ISW_DISK_BLOCK_SIZE;
+
+	to_cpu(isw, FULL);
+
+	return ret;
+}
+
+/*
+ * Group an Intel SW RAID disk into potentially
+ * multiple RAID sets and RAID disks.
+ */
+/* Check state if isw device map. */
+static int _check_map_state(struct lib_context *lc, struct raid_dev *rd,
+			    struct isw_dev *dev)
+{
+	/* FIXME: FAILED_MAP etc. */
+	switch (dev->vol.map.map_state) {
+	case ISW_T_STATE_NORMAL:
+	case ISW_T_STATE_UNINITIALIZED:
+		break;
+
+	default:
+		LOG_ERR(lc, 0, "%s: unsupported map state 0x%x on %s for %s",
+			handler, dev->vol.map.map_state, rd->di->path,
+			(char*) dev->volume);
+	}
+
+	return 1;
+}
+
+/* Create a RAID device to map a volumes segment. */
+static struct raid_dev *_create_rd(struct lib_context *lc, struct raid_dev *rd,
+				   struct isw *isw, struct isw_dev *dev)
+{
+	struct raid_dev *r;
+
+	if (!_check_map_state(lc, rd, dev) ||
+	    !(r = alloc_raid_dev(lc, handler)))
+		return NULL;
+
+	if (!(r->private.ptr = alloc_private(lc, handler, sizeof(*dev))))
+		goto free;
+
+	memcpy(r->private.ptr, dev, sizeof(*dev));
+	if ((r->type = type(r)) == t_undef) {
+		log_err(lc, "%s: RAID type %u not supported",
+			handler, (unsigned int) dev->vol.map.raid_level);
+		goto free;
+	}
+
+        if (!(r->name = name(lc, isw, dev)))
+		goto free;
+
+	r->di = rd->di;
+	r->fmt = rd->fmt;
+	
+	r->offset  = dev->vol.map.pba_of_lba0;
+	r->sectors = dev->vol.map.blocks_per_member;
+
+	goto out;
+
+   free:
+	free_raid_dev(lc, &r);
+   out:
+	return r;
+}
+
+/* Find an Intel RAID set or create it. */
+static void create_rs(struct raid_set *rs, void* private)
+{
+	rs->stride = ((struct isw_dev*) private)->vol.map.blocks_per_strip;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	struct isw *isw = RD(new)->private.ptr;
+	
+	return _get_disk(isw, RD(new)->di) < _get_disk(isw, RD(pos)->di);
+}
+
+/*
+ * rs_group contains the top-level group RAID set (type: t_group) on entry
+ * and shall be returned on success (or NULL on error).
+ */
+static struct raid_set *group_rd(struct lib_context *lc,
+				   struct raid_set *rs_group,
+				   struct raid_dev *rd_meta)
+{
+	unsigned int d;
+	void *private;
+	struct isw *isw = META(rd_meta, isw);
+	struct isw_dev *dev;
+	struct raid_dev *rd;
+	struct raid_set *rs;
+
+	/* Loop the device/volume table. */
+	for (d = 0; d < isw->num_raid_devs; d++) {
+		dev = raiddev(isw, d);
+
+		if (!(rd = _create_rd(lc, rd_meta, isw, dev)))
+			return NULL;
+
+		if (!(rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL,
+				      		  rd, &rs_group->sets,
+						  create_rs, dev))) {
+			free_raid_dev(lc, &rd);
+			return NULL;
+		}
+
+		/* Save and set to enable dev_sort(). */
+		private = rd->private.ptr;
+		rd->private.ptr = isw;
+
+		list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+		/* Restore. */
+		rd->private.ptr = private;
+	}
+
+	return rs_group;
+}
+
+/* Add an Intel SW RAID device to a set */
+static struct raid_set *isw_group(struct lib_context *lc,
+				    struct raid_dev *rd_meta)
+{
+	struct raid_set *rs_group;
+
+	if (T_SPARE(rd_meta))
+		return NULL;
+
+	/*
+	 * Once we get here, an Intel SW RAID disk containing a metadata area
+	 * with a volume table has been discovered by isw_read.
+	 */
+	/* Check if a top level group RAID set already exists. */
+	if (!(rs_group = find_or_alloc_raid_set(lc, rd_meta->name, FIND_TOP,
+				      		rd_meta, LC_RS(lc),
+						NO_CREATE, NO_CREATE_ARG)))
+		return NULL;
+
+	/*
+	 * Add the whole underlying (meta) RAID device to the group set.
+	 * Sorting is no problem here, because RAID sets and devices will
+	 * be created for all the Volumes of an ISW set and those need sorting.
+	 */
+	rd_meta->private.ptr = rd_meta->meta_areas->area;
+	list_add_sorted(lc, &rs_group->devs, &rd_meta->devs, dev_sort);
+	rd_meta->private.ptr = NULL;
+
+	/*
+	 * We need to run through the volume table and create a RAID set and
+	 * RAID devices hanging off it for every volume,
+	 * so that the activate code is happy.
+	 *
+	 * A pointer to the top-level group RAID set
+	 * gets returned or NULL on error.
+	 */
+	return group_rd(lc, rs_group, rd_meta);
+}
+
+/*
+ * Check an Intel SW RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	return ((struct isw_dev*) rd->private.ptr)->vol.map.num_members;
+}
+
+static int check_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_dev *rd, void *context)
+{
+	struct isw_dev *dev = rd->private.ptr;
+
+	/* FIXME: more status checks ? */
+	if (dev->status)
+		LOG_ERR(lc, 0, "%s device for volume \"%s\" broken on %s "
+			"in RAID set \"%s\"",
+			handler, dev->volume, rd->di->path, rs->name);
+
+	return 1;
+}
+
+static int _isw_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL, check_rd, NULL, handler);
+}
+
+static int isw_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return T_GROUP(rs) ? _isw_check(lc, rs) : 0;
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct isw *isw = META(rd, isw);
+	struct isw_disk *disk;
+
+	if (!(disk = get_disk(lc, rd->di, isw)))
+		LOG_ERR(lc, 0, "%s: disk", handler);
+
+	/* Avoid write trashing. */
+	if (S_BROKEN(status(lc, rd)))
+		return 0;
+
+	disk->status &= ~USABLE_DISK;
+	disk->status |= FAILED_DISK;
+
+	return 1;
+}
+
+static struct event_handlers isw_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about an ISW RAID device.
+ */
+static void isw_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int d, i;
+	struct isw *isw = META(rd, isw);
+	struct isw_disk *disk;
+	struct isw_dev *dev;
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	P("sig: \"%*s\"", isw, isw->sig, MAX_SIGNATURE_LENGTH, isw->sig);
+	DP("check_sum: %u", isw, isw->check_sum);
+	DP("mpb_size: %u", isw, isw->mpb_size);
+	DP("family_num: %u", isw, isw->family_num);
+	DP("generation_num: %u", isw, isw->generation_num);
+	DP("reserved[0]: %u", isw, isw->reserved[0]);
+	DP("reserved[1]: %u", isw, isw->reserved[1]);
+	DP("num_disks: %u", isw, isw->num_disks);
+	DP("num_raid_devs: %u", isw, isw->num_raid_devs);
+	DP("fill[0]: %u", isw, isw->fill[0]);
+	DP("fill[1]: %u", isw, isw->fill[1]);
+
+	for (i = 0; i < ISW_FILLERS; i++) {
+		if (isw->filler[i])
+        		P("filler[%i]: %u", isw,
+			  isw->filler[i], i, isw->filler[i]);
+	}
+
+	/* Disk table. */
+	for (d = 0, disk = isw->disk; d < isw->num_disks; d++, disk++) {
+		if (!disk->totalBlocks)
+			continue;
+
+		P("disk[%u].serial: \"%*s\"", isw,
+		  disk->serial, d, MAX_RAID_SERIAL_LEN, disk->serial);
+		P("disk[%u].totalBlocks: %u", isw,
+		  disk->totalBlocks, d, disk->totalBlocks);
+		P("disk[%u].scsiId: 0x%x", isw, disk->scsiId, d, disk->scsiId);
+		P("disk[%u].status: 0x%x", isw, disk->status, d, disk->status);
+		for (i = 0; i < ISW_DISK_FILLERS; i++) {
+			if (disk->filler[i])
+                		P("disk[%u].filler[%u]: %u", isw,
+				  disk->filler[i], d, i, disk->filler[i]);
+		}
+	}
+
+	/* RAID device/volume table. */
+	for (d = 0; d < isw->num_raid_devs; d++) {
+		dev = raiddev(isw, d);
+
+		/* RAID device */
+		P("isw_dev[%u].volume: \"%*s\"", isw,
+		  dev->volume, d, MAX_RAID_SERIAL_LEN, dev->volume);
+		P("isw_dev[%u].SizeHigh: %u", isw,
+		  dev->SizeHigh, d, dev->SizeHigh);
+		P("isw_dev[%u].SizeLow: %u", isw,
+		  dev->SizeLow, d, dev->SizeLow);
+		P("isw_dev[%u].status: 0x%x", isw, dev->status, d, dev->status);
+		P("isw_dev[%u].reserved_blocks: %u", isw,
+		  dev->reserved_blocks, d, dev->reserved_blocks);
+
+		for (i = 0; i < ISW_DEV_FILLERS; i++) {
+			if (dev->filler[i])
+                		P("isw_dev[%u].filler[%u]: %u", isw,
+				  dev->filler[i], d, i, dev->filler[i]);
+		}
+
+		/* RAID volume */
+		for (i = 0; i < 2; i++) {
+			if (dev->vol.reserved[i])
+                		P("isw_dev[%u].vol.reserved[%u]: %u", isw,
+				  dev->vol.reserved[i], d, i,
+				  dev->vol.reserved[i]);
+		}
+
+		P("isw_dev[%u].vol.migr_state: %u", isw,
+		  dev->vol.migr_state, d, dev->vol.migr_state);
+		P("isw_dev[%u].vol.migr_type: %u", isw,
+		  dev->vol.migr_type, d, dev->vol.migr_type);
+		P("isw_dev[%u].vol.dirty: %u", isw,
+		  dev->vol.dirty, d, dev->vol.dirty);
+		P("isw_dev[%u].vol.fill[0]: %u", isw,
+		  dev->vol.fill[0], d, dev->vol.fill[0]);
+
+		for (i = 0; i < 5; i++) {
+			if (dev->vol.filler[i])
+                		P("isw_dev[%u].vol.filler[%u]: %u", isw,
+				  dev->vol.filler[i], d, i,
+				  dev->vol.filler[i]);
+		}
+
+		/* RAID map */
+		P("isw_dev[%u].vol.map.pba_of_lba0: %u", isw,
+		  dev->vol.map.pba_of_lba0, d,
+		  dev->vol.map.pba_of_lba0);
+		P("isw_dev[%u].vol.map.blocks_per_member: %u", isw,
+		  dev->vol.map.blocks_per_member, d,
+		  dev->vol.map.blocks_per_member);
+		P("isw_dev[%u].vol.map.num_data_stripes: %u", isw,
+		  dev->vol.map.num_data_stripes, d,
+		  dev->vol.map.num_data_stripes);
+		P("isw_dev[%u].vol.map.blocks_per_strip: %u", isw,
+		  dev->vol.map.blocks_per_strip, d,
+		  dev->vol.map.blocks_per_strip);
+		P("isw_dev[%u].vol.map.map_state: %u", isw,
+		  dev->vol.map.map_state, d,
+		  dev->vol.map.map_state);
+		P("isw_dev[%u].vol.map.raid_level: %u", isw,
+		  dev->vol.map.raid_level, d,
+		  dev->vol.map.raid_level);
+		P("isw_dev[%u].vol.map.num_members: %u", isw,
+		  dev->vol.map.num_members, d,
+		  dev->vol.map.num_members);
+
+		for (i = 0; i < 3; i++) {
+			if (dev->vol.map.reserved[i])
+                		P("isw_dev[%u].vol.map.reserved[%u]: %u", isw,
+				  dev->vol.map.reserved[i], d, i,
+				  dev->vol.map.reserved[i]);
+		}
+
+		for (i = 0; i < 7; i++) {
+			if (dev->vol.map.filler[i])
+                		P("isw_dev[%u].vol.map.filler[%u]: %u", isw,
+				  dev->vol.map.filler[i], d, i,
+				  dev->vol.map.filler[i]);
+		}
+
+		for (i = 0; i < isw->num_disks; i++) {
+			P("isw_dev[%u].vol.map.disk_ord_tbl[%u]: 0x%x", isw,
+			  dev->vol.map.disk_ord_tbl[i], d, i,
+			  dev->vol.map.disk_ord_tbl[i]);
+		}
+	}
+}
+#endif
+
+static struct dmraid_format isw_format = {
+	.name	= HANDLER,
+	.descr	= "Intel Software RAID",
+	.caps	= "0,1",
+	.format = FMT_RAID,
+	.read	= isw_read,
+	.write	= isw_write,
+	.group	= isw_group,
+	.check	= isw_check,
+	.events	= &isw_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= isw_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_isw(struct lib_context *lc)
+{
+	return register_format_handler(lc, &isw_format);
+}
+
+/*
+ * Set the RAID device contents up derived from the Intel ones.
+ *
+ * This is the first one we get with here and we potentially need to
+ * create many in isw_group() in case of multiple Intel SW RAID devices
+ * on this RAID disk.
+ */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct isw *isw = meta;
+
+	/* Superblock checksum */
+	if (isw->check_sum != _checksum(isw))
+		LOG_ERR(lc, 0, "%s: extended superblock for %s "
+			       "has wrong checksum",
+			handler, di->path);
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = info->u64 >> 9;
+	rd->meta_areas->size = round_up(isw->mpb_size, ISW_DISK_BLOCK_SIZE);
+	rd->meta_areas->area = (void*) isw;
+
+	rd->di = di;
+	rd->fmt = &isw_format;
+
+	rd->offset = ISW_DATAOFFSET;
+	rd->sectors = info->u64 >> 9;
+
+	rd->status = status(lc, rd);
+	rd->type   = t_group;
+
+        return (rd->name = name(lc, isw, NULL)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/isw.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/isw.h
+++ -	2008-02-22 16:57:41.520770000 +0000
@@ -0,0 +1,122 @@
+/*
+ * Copyright (C) 2003,2004,2005 Intel Corporation. 
+ *
+ * dmraid extensions:
+ * Copyright (C) 2004,2005 Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2.1, or (at your option)
+ * any later version.
+ *
+ * You should have received a copy of the GNU General Public License
+ * (for example /usr/src/linux/COPYING); if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ *
+ * Authors: Boji Tony Kannanthanam 
+ *          < boji dot t dot kannanthanam at intel dot com >
+ *          Martins Krikis
+ *          < martins dot krikis at intel dot com >
+ */
+
+/*
+ * Intel Software Raid metadata definitions.
+ */
+
+#ifndef _ISW_H_
+#define _ISW_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+/* Intel metadata offset in bytes */
+#define	ISW_CONFIGOFFSET	((di->sectors - 2) << 9)
+#define	ISW_DATAOFFSET		0	/* Data offset in sectors */
+
+#define MPB_SIGNATURE	     "Intel Raid ISM Cfg Sig. "
+#define MPB_VERSION_RAID2                             "1.2.02"
+#define MAX_SIGNATURE_LENGTH  32
+#define MAX_RAID_SERIAL_LEN   16
+#define ISW_DISK_BLOCK_SIZE  512
+#define TYPICAL_MPBSIZE 1024
+
+/* Disk configuration info. */
+struct isw_disk {
+	int8_t serial[MAX_RAID_SERIAL_LEN];/* 0xD8 - 0xE7 ascii serial number */
+	uint32_t totalBlocks;	/* 0xE8 - 0xEB total blocks */
+	uint32_t scsiId;	/* 0xEC - 0xEF scsi ID */
+	uint32_t status;	/* 0xF0 - 0xF3 */
+#define SPARE_DISK      0x01  /* Spare */
+#define CONFIGURED_DISK 0x02  /* Member of some RaidDev */
+#define FAILED_DISK     0x04  /* Permanent failure */
+#define USABLE_DISK     0x08  /* Fully usable unless FAILED_DISK is set */
+
+#define	ISW_DISK_FILLERS	5
+	uint32_t filler[ISW_DISK_FILLERS]; /* 0xF4 - 0x107 MPB_DISK_FILLERS for future expansion */
+};
+
+/* RAID map configuration infos. */
+struct isw_map {
+	uint32_t pba_of_lba0;		// start address of partition
+	uint32_t blocks_per_member;	// blocks per member
+	uint32_t num_data_stripes;	// number of data stripes
+	uint16_t blocks_per_strip;
+	uint8_t  map_state;		// Normal, Uninitialized, Degraded, Failed
+#define	ISW_T_STATE_NORMAL		0
+#define	ISW_T_STATE_UNINITIALIZED	1
+	uint8_t  raid_level;
+#define	ISW_T_RAID0	0
+#define	ISW_T_RAID1	1
+#define	ISW_T_RAID5	5		// since metadata version 1.2.02 ?
+	uint8_t  num_members;		// number of member disks
+	uint8_t  reserved[3];
+	uint32_t filler[7];		// expansion area
+	uint32_t disk_ord_tbl[1];	/* disk_ord_tbl[num_members],
+					   top byte special */
+} __attribute__ ((packed));
+
+struct isw_vol {
+	uint32_t reserved[2];
+	uint8_t  migr_state;		// Normal or Migrating
+	uint8_t  migr_type;		// Initializing, Rebuilding, ...
+	uint8_t  dirty;
+	uint8_t  fill[1];
+	uint32_t filler[5];
+	struct isw_map map;
+	// here comes another one if migr_state
+} __attribute__ ((packed));
+
+struct isw_dev {
+	uint8_t	volume[MAX_RAID_SERIAL_LEN];
+	uint32_t SizeLow;
+	uint32_t SizeHigh;
+	uint32_t status;	/* Persistent RaidDev status */
+	uint32_t reserved_blocks; /* Reserved blocks at beginning of volume */
+#define	ISW_DEV_FILLERS	12
+	uint32_t filler[ISW_DEV_FILLERS];
+	struct isw_vol vol;
+} __attribute__ ((packed));
+
+struct isw {
+	int8_t sig[MAX_SIGNATURE_LENGTH];/* 0x0 - 0x1F */
+	uint32_t check_sum;		/* 0x20 - 0x23  MPB Checksum */
+	uint32_t mpb_size;		/* 0x24 - 0x27 Size of MPB */
+	uint32_t family_num;		/* 0x28 - 0x2B Checksum from first time this config was written */
+	/* 0x2C - 0x2F  Incremented each time this array's MPB is written */
+	uint32_t generation_num;
+	uint32_t reserved[2];		/* 0x30 - 0x37 */
+	uint8_t num_disks;		/* 0x38 Number of configured disks */
+	uint8_t num_raid_devs;		/* 0x39 Number of configured volumes */
+	uint8_t fill[2];		/* 0x3A - 0x3B */
+#define	ISW_FILLERS	39
+	uint32_t filler[ISW_FILLERS];	/* 0x3C - 0xD7 RAID_MPB_FILLERS */
+	struct isw_disk disk[1];	/* 0xD8 diskTbl[numDisks] */
+	// here comes isw_dev[num_raid_devs]
+} __attribute__ ((packed));
+
+#endif
+
+int register_isw(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/jm.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/jm.c
+++ -	2008-02-22 16:57:41.607120000 +0000
@@ -0,0 +1,379 @@
+/*
+ * Copyright (C) 2006  Heinz Mauelshagen, Red Hat GmbH.
+ *                     All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * JMicron ATARAID metadata format handler.
+ */
+#define	HANDLER "jmicron"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "jm.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/* RAID set name */
+static int member(struct jm *jm);
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	struct jm *jm = META(rd, jm);
+	char buf[2], *ret, *name = (char *) jm->name;
+
+	/* Name always 0 terminated ? */
+	if ((len = strlen(name)) > JM_NAME_LEN)
+		len = JM_NAME_LEN;
+
+	len += sizeof(HANDLER) + 2;
+	if (jm->mode == JM_T_RAID01)
+		len++;
+
+	if ((ret = dbg_malloc(len))) {
+		if (jm->mode == JM_T_RAID01 && subset)
+			sprintf(buf, "-%u", member(jm) / 2);
+		else
+			*buf = 0;
+
+		sprintf(ret, "%s_%s%s", HANDLER, name, buf);
+	}
+
+	return ret;
+}
+
+/*
+ * Retrieve status of device.
+ * FIXME: is this sufficient to cover all state ?
+ */
+static enum status status(struct jm *jm)
+{
+	return jm->attribute & ~(JM_MOUNT|JM_BOOTABLE|JM_BADSEC|JM_ACTIVE|JM_UNSYNC|JM_NEWEST) ? s_broken : s_ok;
+}
+
+/* Mapping of JM types to generic types */
+static struct types types[] = {
+	{ JM_T_JBOD, t_linear},
+	{ JM_T_RAID0, t_raid0},
+	{ JM_T_RAID01, t_raid1},
+	{ JM_T_RAID1, t_raid1},
+	{ 0, t_undef}
+};
+
+/* Neutralize disk type */
+static enum type type(struct jm *jm)
+{
+	return rd_type(types, (unsigned int) jm->mode);
+}
+
+/* Calculate checksum on metadata */
+static int checksum(struct jm *jm)
+{
+	int count = 64;
+	uint16_t *p = (uint16_t*) jm, sum = 0;
+
+        while (count--)
+		sum += *p++;
+
+	return sum;
+}
+
+static inline unsigned int segment(uint32_t m)
+{
+	return (unsigned int) (m & JM_SEG_MASK);
+}
+
+static inline unsigned int disk(unsigned int m)
+{
+	return (unsigned int) (m & JM_HDD_MASK);
+}
+
+static int member(struct jm *jm)
+{
+	unsigned int i = JM_MEMBERS;
+
+	while (i--) {
+		if (disk(jm->member[i]) == disk(jm->identity))
+			return i;
+	}
+
+	return -1;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return member(META(RD(new), jm)) < member(META(RD(pos), jm));
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return member(META(RD_RS(RS(pos)), jm)) > 1;
+}
+
+static unsigned int stride(unsigned int shift)
+{
+	return 1 << (shift + 1);
+}
+
+static void super_created(struct raid_set *super, void *private)
+{
+	super->type   = t_raid0;
+	super->stride = stride(META((private), jm)->block);
+}
+
+/*
+ * Group the RAID disk into a JM set.
+ *
+ * Check device hierarchy and create super set appropriately.
+ */
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct jm *jm = META(rd, jm);
+
+	if (!init_raid_set(lc, rs, rd, stride(jm->block), jm->mode, handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (jm->mode) {
+	case JM_T_JBOD:
+	case JM_T_RAID0:
+	case JM_T_RAID1:
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case JM_T_RAID01:
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+/*
+ * Add a JMicron RAID device to a set.
+ */
+static struct raid_set *jm_group(struct lib_context *lc,
+				       struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/*
+ * Read a JMicron RAID device.
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	unsigned int i;
+	struct jm *jm = meta;
+
+	CVT16(jm->version);
+	CVT16(jm->checksum);
+	CVT32(jm->identity);
+
+	CVT32(jm->segment.base);
+	CVT32(jm->segment.range);
+	CVT16(jm->segment.range2);
+
+	CVT16(jm->attribute);
+
+	for (i = 0; i < JM_SPARES; i++)
+		CVT32(jm->spare[i]);
+
+	for (i = 0; i < JM_MEMBERS; i++)
+		CVT32(jm->member[i]);
+}
+#endif
+
+/* Magic check. */
+static int is_jm(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct jm *jm = meta;
+
+	return !strncmp((const char*) jm->signature,
+			JM_SIGNATURE, JM_SIGNATURE_LEN)
+	       && !checksum(jm);
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *jm_read(struct lib_context *lc,
+				      struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct jm), JM_CONFIGOFFSET,
+			     to_cpu, is_jm, NULL, setup_rd, handler);
+}
+
+/*
+ * Write a JMicron RAID device.
+ */
+static int jm_write(struct lib_context *lc,
+			struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct jm *jm = META(rd, jm);
+
+	to_disk(jm);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(jm);
+#endif
+	return ret;
+}
+
+/*
+ * Check a JMicron RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	unsigned int r = JM_MEMBERS;
+	struct jm *jm = META(rd, jm);
+
+	while (r-- && !jm->member[r]);
+
+	return ++r;
+}
+
+static int jm_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL,
+			      NO_CHECK_RD, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct jm *jm = META(rd, jm);
+
+	/* Avoid write trashing. */
+	if (S_BROKEN(status(jm)))
+		return 0;
+
+	jm->checksum = 1; /* FIXME: how to flag a JMicron disk bad? */
+
+	return 1;
+}
+
+static struct event_handlers jm_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about an JM RAID device.
+ */
+static void jm_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int i;
+	struct jm *jm = META(rd, jm);
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	P("signature: %c%c", jm, jm->signature,
+	  jm->signature[0], jm->signature[1]);
+	P("version: %u%u", jm, jm->version,
+	  JM_MAJOR_VERSION(jm), JM_MINOR_VERSION(jm));
+	DP("checksum: %u", jm, jm->checksum);
+	DP("identity: 0x%x", jm, jm->identity);
+	DP("base: %u", jm, jm->segment.base);
+	DP("range: %u", jm, jm->segment.range);
+	DP("range2: %u", jm, jm->segment.range2);
+	DP("name: \"%s\"", jm, jm->name);
+	DP("name: %u", jm, jm->mode);
+	DP("block: %u", jm, jm->block);
+	DP("attribute: %u", jm, jm->attribute);
+	for (i = 0; i < JM_SPARES; i++)
+		P2("spare[%d]: 0x%x", jm, i, jm->spare[i]);
+	for (i = 0; i < JM_MEMBERS; i++)
+		P2("member[%d]: 0x%x", jm, i, jm->member[i]);
+}
+#endif
+
+static struct dmraid_format jm_format = {
+	.name	= HANDLER,
+	.descr	= "JMicron ATARAID",
+	.caps	= "S,0,1",
+	.format = FMT_RAID,
+	.read	= jm_read,
+	.write	= jm_write,
+	.group	= jm_group,
+	.check	= jm_check,
+	.events	= &jm_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= jm_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_jm(struct lib_context *lc)
+{
+	return register_format_handler(lc, &jm_format);
+}
+
+/* Calculate RAID device size in sectors depending on RAID type. */
+static inline uint64_t sectors(struct jm *jm)
+{
+	/* range * 32MB[sectors] + range2 */
+	return jm->segment.range * 32*2048 + jm->segment.range2;
+}
+
+/* Set the RAID device contents up derived from the JMicron ones */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct jm *jm = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = JM_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*jm);
+	rd->meta_areas->area = (void*) jm;
+
+	rd->di = di;
+	rd->fmt = &jm_format;
+
+	rd->status = status(jm);
+	rd->type   = type(jm);
+
+	rd->offset = jm->segment.base;
+	if (!(rd->sectors = sectors(jm)))
+		return log_zero_sectors(lc, di->path, handler);
+
+	return (rd->name = name(lc, rd, 1)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/jm.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/jm.h
+++ -	2008-02-22 16:57:41.712131000 +0000
@@ -0,0 +1,77 @@
+/*
+ * Copyright (C) 2006  Heinz Mauelshagen, Red Hat GmbH.
+ *                     All rights reserved.
+ *
+ * Based on metadata specs kindly provided by JMicron
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_JMICRON_H_
+#define	_JMICRON_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+/* JMicron config data sector offset off end of disk */
+#define JM_CONFIGOFFSET	((di->sectors - 1) << 9)
+
+/* Ondisk metadata for JMicron ATARAID */
+struct jm {
+#define JM_SIGNATURE "JM"
+#define JM_SIGNATURE_LEN	2
+	int8_t		signature[JM_SIGNATURE_LEN];	/* 0x0 - 0x01 */
+
+	uint16_t	version;	/* 0x03 - 0x04 JMicron version */
+#define	JM_MINOR_VERSION(jm)	(jm->version & 0xFF)
+#define	JM_MAJOR_VERSION(jm)	(jm->version >> 8)
+
+	uint16_t	checksum;	/* 0x04 - 0x05 */
+	uint8_t		filler[10];
+
+	uint32_t	identity;	/* 0x10 - 0x13 */
+
+	struct {
+		uint32_t	base;	/* 0x14 - 0x17 */
+		uint32_t	range;	/* 0x18 - 0x1B range */
+		uint16_t	range2;	/* 0x1C - 0x1D range2 */
+	} segment;
+
+#define	JM_NAME_LEN	16
+	int8_t		name[JM_NAME_LEN];	/* 0x20 - 0x2F */
+
+	uint8_t		mode;		/* 0x30 RAID level */
+#define	JM_T_RAID0	0
+#define	JM_T_RAID1	1
+#define	JM_T_RAID01	2	/* RAID 0+1 (striped with mirrors underneath) */
+#define	JM_T_JBOD	3
+#define	JM_T_RAID5	5
+
+	uint8_t		block;		/* 0x31 stride size (2=4K, 3=8K, ...) */
+	uint16_t	attribute;	/* 0x32 - 0x33 */ 
+#define	JM_MOUNT	0x01
+#define	JM_BOOTABLE	0x02
+#define	JM_BADSEC	0x03
+#define	JM_ACTIVE	0x05
+#define	JM_UNSYNC	0x06
+#define	JM_NEWEST	0x07
+
+	uint8_t		filler1[4];
+
+#define	JM_SPARES	2
+#define	JM_MEMBERS	8
+	uint32_t	spare[JM_SPARES];	/* 0x38 - 0x3F */
+	uint32_t	member[JM_MEMBERS];	/* 0x40 - 0x5F */
+#define	JM_HDD_MASK	0xFFFFFFF0
+#define	JM_SEG_MASK	0x0F
+
+	uint8_t		filler2[0x20];
+} __attribute__ ((packed));
+
+#endif
+
+int register_jm(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/lsi.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/lsi.c
+++ -	2008-02-22 16:57:41.827473000 +0000
@@ -0,0 +1,359 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *			    All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * LSI Logic MegaRAID (and MegaIDE ?) ATARAID metadata format handler.
+ *
+ * Needs more metadata reengineering and grouping logic coding.
+ *
+ */
+
+#define	HANDLER	"lsi"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "lsi.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/* Make up RAID device name. */
+/* FIXME: senseful name ;) */
+static unsigned int get_disk_slot(struct lsi *lsi)
+{
+	return lsi->set_number * 2 + lsi->disk_number;
+}
+
+static struct lsi_disk *get_disk(struct lsi *lsi)
+{
+	return lsi->disks + get_disk_slot(lsi);
+}
+
+static size_t _name(struct lsi *lsi, char *str, size_t len, unsigned int subset)
+{
+	return snprintf(str, len,
+			subset ? "lsi_%u%u-%u" : "lsi_%u%u",
+ 			lsi->set_id, lsi->set_number,
+			(get_disk(lsi))->raid10_mirror);
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret;
+	struct lsi *lsi = META(rd, lsi);
+
+	subset = subset && (lsi->type == LSI_T_RAID10);
+	if ((ret = dbg_malloc((len = _name(lsi, NULL, 0, subset) + 1)))) {
+		_name(lsi, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, strlen(ret) - HANDLER_LEN -
+			 (subset ? 2 : 0));
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/* Mapping of LSI Logic types to generic types */
+static struct types types[] = {
+	{ LSI_T_RAID0, t_raid0 },
+	{ LSI_T_RAID1, t_raid1 },
+	{ LSI_T_RAID10, t_raid0 },
+        { 0, t_undef}
+};
+
+/* Neutralize disk type */
+static enum type type(struct lsi *lsi)
+{
+	return rd_type(types, (unsigned int) lsi->type);
+}
+
+/* LSI device status. */
+/* FIXME: add flesh. */
+static int status(struct lsi *lsi)
+{
+	return s_ok;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	struct lsi *p = META(RD(pos), lsi), *n = META(RD(new), lsi);
+
+	switch (n->type) {
+	case LSI_T_RAID10:
+		return (get_disk(n))->raid10_stripe < 
+		       (get_disk(p))->raid10_stripe;
+
+	default: /* RAID0 + RAID01 */
+		return get_disk_slot(n) < get_disk_slot(p);
+	}
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	struct lsi *p = META(RD_RS(pos), lsi), *n = META(RD_RS(new), lsi);
+
+	return n->disks[get_disk_slot(n)].raid10_mirror <
+	       p->disks[get_disk_slot(p)].raid10_mirror;
+}
+
+/*
+ * Read an LSI Logic RAID device
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct lsi *lsi = meta;
+	struct lsi_disk *disk;
+
+	CVT16(lsi->stride);
+
+	for (disk = lsi->disks; disk < &lsi->disks[LSI_MAX_DISKS]; disk++) {
+		CVT16(disk->magic_0);
+		CVT16(disk->magic_1);
+	}
+
+	CVT32(lsi->set_id);
+}
+#endif
+
+static int is_lsi(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	return !strncmp((const char*) ((struct lsi *) meta)->magic_name,
+	       LSI_MAGIC_NAME, LSI_MAGIC_NAME_LEN);
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *lsi_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct lsi), LSI_CONFIGOFFSET, 
+			     to_cpu, is_lsi, NULL, setup_rd, handler);
+}
+
+/*
+ * Write a LSI Logic RAID device.
+ */
+static int lsi_write(struct lib_context *lc, struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct lsi *lsi = META(rd, lsi);
+
+	to_disk(lsi);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(lsi);
+#endif
+	return ret;
+}
+
+/*
+ * Group the RAID disk into a set.
+ *
+ * FIXME: this needs more work together with the metadata reengineering.
+ */
+static void super_created(struct raid_set *ss, void *private)
+{
+	ss->type   = t_raid1;
+	ss->stride = META(private, lsi)->stride;
+}
+
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct lsi *lsi = META(rd, lsi);
+
+	if (!init_raid_set(lc, rs, rd, lsi->stride, type(lsi), handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (lsi->type) {
+	case LSI_T_RAID0:
+	case LSI_T_RAID1:
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+		break;
+
+	case LSI_T_RAID10:
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+static struct raid_set *lsi_group(struct lib_context *lc,
+				    struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/* Figure total number of disks depending on RAID type. */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	switch ((META(rd, lsi))->type) {
+	case LSI_T_RAID10:
+		return 4;
+
+	case LSI_T_RAID0:
+	case LSI_T_RAID1:
+		return 2;
+	}
+
+	return 0;
+}
+
+/*
+ * Check an LSI RAID set.
+ *
+ * FIXME: more sanity checks!!!
+ */
+static int lsi_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL,
+			      NO_CHECK_RD, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct lsi *lsi = META(rd, lsi);
+
+	/* Avoid write trashing. */
+	if (status(lsi) & s_broken)
+		return 0;
+
+	// FIXME: lsi->? = BAD;
+
+	return 1;
+}
+
+static struct event_handlers lsi_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/* Log native information about an LSI Logic RAID device. */
+static void lsi_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int i;
+	struct lsi *lsi = META(rd, lsi);
+	struct lsi_disk *disk;
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	DP("magic_name: %s", lsi, lsi->magic_name);
+	P("dummy: %u, 0x%x", lsi, lsi->dummy, lsi->dummy, lsi->dummy);
+	DP("seqno: %u", lsi, lsi->seqno);
+	P("dummy2: %u, 0x%x", lsi, lsi->dummy2, lsi->dummy2, lsi->dummy2);
+	P("dummy3: %u, 0x%x", lsi, lsi->dummy3, lsi->dummy3, lsi->dummy3);
+	DP("type: %u", lsi, lsi->type);
+	P("dummy4: %u, 0x%x", lsi, lsi->dummy4, lsi->dummy4, lsi->dummy4);
+	DP("stride: %u", lsi, lsi->stride);
+
+	for (disk = lsi->disks, i = 0; i < LSI_MAX_DISKS; disk++, i++) {
+		P("disks[%u].raid10_stripe: %u", lsi, disk, i,
+		  disk->raid10_stripe);
+		P("disks[%u].raid10_mirror: %u", lsi, disk, i,
+		  disk->raid10_mirror);
+		P("disks[%u].unknown: %u, 0x%x", lsi, disk, i,
+		  disk->unknown, disk->unknown);
+		P("disks[%u].magic_0: 0x%x, %x, %x", lsi,
+		  disk->magic_0, i, disk->magic_0,
+		  (unsigned char) (((char*) &disk->magic_0)[0]),
+		  (unsigned char) (((char*) &disk->magic_0)[1]));
+		P("disks[%u].magic_1: 0x%x, %x, %x", lsi,
+		  disk->magic_1, i, disk->magic_1,
+		  (unsigned char) (((char*) &disk->magic_1)[0]),
+		  (unsigned char) (((char*) &disk->magic_1)[1]));
+		P("disks[%u].disk_number: %u", lsi, disk->disk_number,
+		  i, disk->disk_number);
+		P("disks[%u].set_number: %u", lsi, disk->set_number,
+		  i, disk->set_number);
+		P("disks[%u].unknown1: %u, 0x%u", lsi, disk->unknown1,
+		  i, disk->unknown1, disk->unknown1);
+	}
+
+	DP("disk_number: %u", lsi, lsi->disk_number);
+	DP("set_number: %u", lsi, lsi->set_number);
+	DP("set_id: %u", lsi, lsi->set_id);
+}
+#endif
+
+static struct dmraid_format lsi_format = {
+	.name	= HANDLER,
+	.descr	= "LSI Logic MegaRAID",
+	.caps	= "0,1,10",
+	.format = FMT_RAID,
+	.read	= lsi_read,
+	.write	= lsi_write,
+	.group	= lsi_group,
+	.check	= lsi_check,
+	.events	= &lsi_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= lsi_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_lsi(struct lib_context *lc)
+{
+	return register_format_handler(lc, &lsi_format);
+}
+
+/* Set the RAID device contents up derived from the LSI ones */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct lsi *lsi = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = LSI_CONFIGOFFSET >> 9;
+	rd->meta_areas->size   = sizeof(*lsi);
+	rd->meta_areas->area   = (void*) lsi;
+
+        rd->di = di;
+	rd->fmt = &lsi_format;
+
+	rd->status = status(lsi);
+	rd->type   = type(lsi);
+
+	rd->offset = LSI_DATAOFFSET;
+	/* FIXME: propper size ? */
+	rd->sectors = rd->meta_areas->offset;
+
+        return (rd->name = name(lc, rd, 1)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/lsi.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/lsi.h
+++ -	2008-02-22 16:57:41.943455000 +0000
@@ -0,0 +1,71 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_LSI_H_
+#define	_LSI_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+/*
+ * FIXME: this needs more reengineering still.
+ */
+
+#include <stdint.h>
+
+#define	LSI_CONFIGOFFSET	((di->sectors - 1) << 9)
+#define	LSI_DATAOFFSET		0
+
+struct lsi_disk {
+	uint16_t raid10_stripe:4;
+	uint16_t raid10_mirror:4;
+	uint16_t unknown:8;
+	uint16_t magic_0;
+	uint16_t magic_1;
+	uint8_t disk_number;
+	uint8_t set_number;
+	uint8_t unknown1[8];
+} __attribute__ ((packed));
+
+struct lsi {
+#define	LSI_MAGIC_NAME	"$XIDE$"
+#define	LSI_MAGIC_NAME_LEN	(sizeof(LSI_MAGIC_NAME) - 1)
+	uint8_t magic_name[LSI_MAGIC_NAME_LEN];	/* 0x0 - 0x05 */
+	uint8_t  dummy;		/* 0x06 */
+	uint8_t  seqno;		/* 0x07 */
+	uint32_t dummy2;		/* 0x08 - 0x0B */
+	uint32_t dummy3;		/* 0x0C - 0x0F */
+	uint8_t  type;			/* 0x10 */
+#define LSI_T_RAID0	1
+#define LSI_T_RAID1	2
+#define LSI_T_RAID10	3
+
+	uint8_t dummy4;		/* 0x11 */
+	uint16_t stride;		/* 0x12 - 0x13 */
+	uint8_t filler[0x20-0x14];	/* 0x14 - 0x1F */
+#define	LSI_MAX_DISKS	4
+	struct lsi_disk disks[LSI_MAX_DISKS];/* 0x20 - 0x5F */
+#define	LSI_DISK(lsi)		(lsi->set_number * 2 + lsi->disk_number)
+#define LSI_MAGIC_0(lsi)	(lsi->disks[LSI_DISK(lsi)].magic_0)
+#define LSI_MAGIC_1(lsi)	(lsi->disks[LSI_DISK(lsi)].magic_1)
+#define LSI_DISK_NUMBER(lsi)	(lsi->disks[LSI_DISK(lsi)].disk_number)
+#define LSI_SET_NUMBER(lsi)	(lsi->disks[LSI_DISK(lsi)].set_number)
+#undef	LSI_DISK
+
+	uint8_t filler1[0x1F0-0x60];	/* 0x70 - 0x1EF */
+
+	uint8_t disk_number;		/* 0x1F0 */
+	uint8_t set_number;		/* 0x1F1 */
+	uint32_t set_id;		/* 0x1F2 - 0x1F5 */
+
+	uint8_t filler2[0x200-0x1F6];	/* 0x1F6 - 0x200 */
+} __attribute__ ((packed));
+#endif
+
+int register_lsi(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/nv.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/nv.c
+++ -	2008-02-22 16:57:42.023077000 +0000
@@ -0,0 +1,432 @@
+/*
+ * Copyright (C) 2004      NVidia Corporation. All rights reserved.
+ * Copyright (C) 2004,2005 Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * NVidia NVRAID metadata format handler.
+ */
+#define	HANDLER "nvidia"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "nv.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/*
+ * Make up RAID device name from array signature.
+ *
+ * The subset parameter indicates the requirement to create
+ * name suffixes in case the RAID set is hierarchical.
+ */
+static unsigned int _subset(struct nv *nv)
+{
+	return nv->unitNumber >= nv->array.stripeWidth;
+}
+
+static size_t _name(struct nv *nv, char *str, size_t len, unsigned int subset)
+{
+	unsigned int i = NV_SIGNATURES;
+	uint32_t sum = 0;
+
+	while (i--)
+		sum += nv->array.signature[i];
+
+	return snprintf(str, len, subset ? "%s_%.8x-%u" : "%s_%.8x",
+			handler, sum, _subset(nv));
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret;
+	struct nv *nv = META(rd, nv);
+
+	subset &= NVRAID_1_0(nv);
+	if ((ret = dbg_malloc((len = _name(nv, NULL, 0, subset) + 1)))) {
+		_name(nv, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN - subset - 1);
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/* Mapping of nv types to generic types */
+static struct types types[] = {
+	{ NV_LEVEL_JBOD, t_linear },
+	{ NV_LEVEL_0, t_raid0 },
+	{ NV_LEVEL_1, t_raid1 },
+	{ NV_LEVEL_1_0, t_raid0 },	/* Treat as 0 here, add mirror later */
+	{ NV_LEVEL_3, t_raid4 },
+	{ NV_LEVEL_5_SYM, t_raid5_ls },
+        { NV_LEVEL_UNKNOWN, t_spare},	/* FIXME: UNKNOWN = spare ? */
+	/* FIXME: The ones below don't really map to anything ?? */
+	{ NV_LEVEL_10, t_undef },
+	{ NV_LEVEL_5, t_undef },    /* Asymmetric RAID 5 is not used */
+};
+
+static enum status status(struct nv *nv)
+{
+	if (NV_BROKEN(nv))
+		return s_broken;
+	
+	switch(nv->array.raidJobCode) {
+	case NV_IDLE:
+		return s_ok;
+
+	case NV_SCDB_INIT_RAID:
+	case NV_SCDB_SYNC_RAID:
+		return s_nosync;
+
+	case NV_SCDB_REBUILD_RAID:
+	case NV_SCDB_UPGRADE_RAID:
+		return s_inconsistent;
+	}
+
+	return s_broken; 
+}
+
+/* Neutralize disk type using generic metadata type mapping function. */
+static enum type type(struct nv *nv)
+{
+	uint8_t stripeWidth = nv->array.stripeWidth;
+
+	/*
+	 * FIXME: is there a direct way to decide what
+	 * 	  a spare is (eg, NV_LEVEL_UNKNOWN) ?
+	 */
+	switch(NV_RAIDLEVEL(nv)) {
+	case NV_LEVEL_1_0:
+	case NV_LEVEL_10:
+	case NV_LEVEL_1:
+		stripeWidth *= 2;
+		break;
+
+	case NV_LEVEL_5_SYM:
+		stripeWidth += 1;
+		break;
+		
+			default:
+		break;
+	}
+
+	if (nv->array.totalVolumes >= stripeWidth &&
+	    nv->unitNumber         >= stripeWidth)
+		return t_spare;
+
+	return rd_type(types, (unsigned int) NV_RAIDLEVEL(nv));
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD(new), nv))->unitNumber <
+	       (META(RD(pos), nv))->unitNumber;
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return _subset((META(RD_RS(RS(new)), nv))) <
+	       _subset((META(RD_RS(RS(pos)), nv)));
+}
+
+/*
+ * Read a Template RAID device
+ */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct nv *nv = meta;
+	unsigned int i = NV_SIGNATURES;
+	struct nv_array_base *array = &nv->array;
+
+	CVT32(nv->size);
+	CVT32(nv->chksum);
+	CVT16(nv->version);
+	CVT32(nv->capacity);
+	CVT32(nv->sectorSize);
+	CVT32(nv->unitFlags);
+	CVT32(array->version);
+
+	while (i--);
+		CVT32(array->signature[i]);
+
+	CVT32(array->raidLevel);
+	CVT32(array->stripeBlockSize);
+	CVT32(array->stripeBlockByteSize);
+	CVT32(array->stripeBlockPower);
+	CVT32(array->stripeMask);
+	CVT32(array->stripeSize);
+	CVT32(array->stripeByteSize);
+	CVT32(array->raidJobMark);
+	CVT32(array->originalLevel);
+	CVT32(array->originalCapacity);
+	CVT32(array->flags);
+}
+#endif
+
+/* Check the metadata checksum. */
+static int checksum(struct nv *nv)
+{
+	uint32_t sum = 0;
+	unsigned int s = nv->size;
+
+	if (s != sizeof(*nv) / sizeof(sum))
+		return 0;
+
+	while (s--) 
+		sum += ((uint32_t*) nv)[s];
+
+	/* Ignore chksum member itself. */
+	return nv->chksum - sum == nv->chksum;
+}
+
+static int is_nv(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct nv *nv = meta;
+
+	if (strncmp((char*) nv->vendor, NV_ID_STRING, sizeof(NV_ID_STRING) - 1))
+		return 0;
+
+	if (checksum(nv))
+		return 1;
+
+	LOG_ERR(lc, 0, "%s: bad checksum on %s", handler, di->path);
+}
+	
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *nv_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct nv), NV_CONFIGOFFSET,
+			     to_cpu, is_nv, NULL, setup_rd, handler);
+}
+
+/* Write private RAID metadata to device */
+static int nv_write(struct lib_context *lc, struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct nv *nv = META(rd, nv);
+
+	to_disk(nv);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(nv);
+#endif
+	return ret;
+}
+
+static void super_created(struct raid_set *ss, void *private)
+{
+	ss->type   = t_raid1;
+	ss->stride = META(private, nv)->array.stripeBlockSize;
+}
+
+/* FIXME: handle spares in mirrors and check that types are correct. */
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct nv *nv = META(rd, nv);
+
+	if (!init_raid_set(lc, rs, rd, nv->array.stripeBlockSize,
+			   NV_RAIDLEVEL(nv), handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (NV_RAIDLEVEL(nv)) {
+	case NV_LEVEL_JBOD:
+	case NV_LEVEL_0:
+	case NV_LEVEL_1:
+	case NV_LEVEL_5_SYM:
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case NV_LEVEL_1_0:
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+/* Add an NVidia RAID device to a set. */
+static struct raid_set *nv_group(struct lib_context *lc,
+				       struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	/* Spares are grouped with sets. */
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/*
+ * Check integrity of an NVidia RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	struct nv *nv = META(rd, nv);
+
+	return nv->array.totalVolumes / (NVRAID_1_0(nv) ? 2 : 1);
+}
+
+static int nv_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL,
+			      NO_CHECK_RD, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct nv *nv = META(rd, nv);
+
+	/* Avoid write trashing. */
+	if (status(nv) & s_broken)
+		return 0;
+
+	NV_SET_BROKEN(nv);
+	log_err(lc, "%s: signature recalculation missing!", handler);
+
+	return 1;
+}
+
+static struct event_handlers nv_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about the RAID device.
+ */
+static void nv_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int i, j;
+#define	LEN	NV_PRODUCTIDS + 1
+	char buffer[LEN];
+	struct nv *nv = META(rd, nv);
+	struct nv_array_base *a = &nv->array;
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	P("%*s", nv, nv->vendor, NV_ID_LENGTH, nv->vendor);
+	DP("size: %u", nv, nv->size);
+	DP("chksum: %u", nv, nv->chksum);
+	DP("version: %u", nv, nv->version);
+	DP("unitNumber: %u", nv, nv->unitNumber);
+	DP("reserved: %u", nv, nv->reserved);
+	DP("capacity: %u", nv, nv->capacity);
+	DP("sectorSize: %u", nv, nv->sectorSize);
+
+	for (i = 0; i < NV_PRODUCTIDS; i++)
+		buffer[i] = nv->productID[i];
+
+	buffer[i] = '\0';
+	P("productID: %s", nv, nv->productID, buffer);
+
+	for (i = j = 0; i < NV_PRODUCTREVISIONS; i++) {
+		if (nv->productRevision[i])
+			buffer[j++] = nv->productRevision[i];
+	}
+
+	buffer[j] = '\0';
+	P("productRevision: %s", nv, nv->productRevision, buffer);
+	DP("unitFlags: %u", nv, nv->unitFlags);
+	DP("array->version: %u", nv, a->version);
+
+	for (i = 0; i < NV_SIGNATURES; i++)
+		P("array->signature[%d]: %u",
+		  nv, a->signature[i], i, a->signature[i]);
+
+	DP("array->raidJobCode: %u", nv, a->raidJobCode);
+	DP("array->stripeWidth: %u", nv, a->stripeWidth);
+	DP("array->totalVolumes: %u", nv, a->totalVolumes);
+	DP("array->originalWidth: %u", nv, a->originalWidth);
+	DP("array->raidLevel: %u", nv, a->raidLevel);
+	DP("array->stripeBlockSize: %u", nv, a->stripeBlockSize);
+	DP("array->stripeBlockByteSize: %u", nv, a->stripeBlockByteSize);
+	DP("array->stripeBlockPower: %u", nv, a->stripeBlockPower);
+	DP("array->stripeMask: %u", nv, a->stripeMask);
+	DP("array->stripeSize: %u", nv, a->stripeSize);
+	DP("array->stripeByteSize: %u", nv, a->stripeByteSize);
+	DP("array->raidJobMark %u", nv, a->raidJobMark);
+	DP("array->originalLevel %u", nv, a->originalLevel);
+	DP("array->originalCapacity %u", nv, a->originalCapacity);
+	DP("array->flags 0x%x", nv, a->flags);
+}
+#endif
+
+static struct dmraid_format nv_format = {
+	.name	= HANDLER,
+	.descr	= "NVidia RAID",
+	.caps	= "S,0,1,10,5",
+	.format = FMT_RAID,
+	.read	= nv_read,
+	.write	= nv_write,
+	.group	= nv_group,
+	.check	= nv_check,
+	.events	= &nv_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= nv_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_nv(struct lib_context *lc)
+{
+	return register_format_handler(lc, &nv_format);
+}
+
+/* Set the RAID device contents up derived from the NV ones */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct nv *nv = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = NV_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*nv);
+	rd->meta_areas->area = (void*) nv;
+
+        rd->di = di;
+	rd->fmt = &nv_format;
+
+	rd->status = status(nv);
+	rd->type   = type(nv);
+
+	rd->offset = NV_DATAOFFSET;
+	rd->sectors = rd->meta_areas->offset;
+
+        return (rd->name = name(lc, rd, 1)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/nv.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/nv.h
+++ -	2008-02-22 16:57:42.118627000 +0000
@@ -0,0 +1,114 @@
+/*
+ * Copyright (C) 2004,2005  NVidia Corporation. All rights reserved.
+ *
+ * dmraid extensions:
+ * Copyright (C) 2004,2005 Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * You should have received a copy of the GNU General Public License
+ * (for example /usr/src/linux/COPYING); if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ *
+ */
+
+#ifndef	_NV_H_
+#define	_NV_H_
+
+#ifdef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+#define	NV_CONFIGOFFSET		((di->sectors - 2) << 9)
+#define	NV_DATAOFFSET		0
+
+#define NV_ID_LENGTH		8
+#define NV_ID_STRING		"NVIDIA"
+#define NV_VERSION		100 
+#define NV_SECTOR_SIZE		512
+#define NV_PRODUCT_ID_LEN	16 /* Product ID size in bytes */
+
+typedef uint32_t lba_t;
+
+/* Array info */
+struct nv_array_base {
+	uint32_t version;	/* Version of this struct               */
+				/* 0x640000 + sizeof(nv_array_base) */
+#define NV_SIGNATURES	4
+	uint32_t signature[NV_SIGNATURES]; /* Unique signature for array */
+
+	uint8_t  raidJobCode;	/* State of array */ 
+#define NV_IDLE			0
+#define NV_SCDB_INIT_RAID	2
+#define NV_SCDB_REBUILD_RAID	3
+#define NV_SCDB_UPGRADE_RAID	4
+#define NV_SCDB_SYNC_RAID	5
+
+	uint8_t  stripeWidth;	/* Array stripe width */
+	uint8_t  totalVolumes;	/* Total # of disks in array, including spare */
+	uint8_t  originalWidth;	/* Stripe width before morph */
+
+	uint32_t raidLevel;	/* Array RAID level */
+#define	NV_RAIDLEVEL(nv)	((nv)->array.raidLevel)
+#define NV_LEVEL_UNKNOWN		0x00
+#define NV_LEVEL_JBOD			0xFF
+#define NV_LEVEL_0			0x80
+#define NV_LEVEL_1			0x81
+#define NV_LEVEL_3			0x83
+#define NV_LEVEL_5			0x85
+#define NV_LEVEL_10			0x8a
+#define NV_LEVEL_1_0			0x8180
+#define NVRAID_1_0(nv)	(NV_RAIDLEVEL((nv)) == NV_LEVEL_1_0)
+#define NV_LEVEL_5_SYM_FLAG		0x10
+#define NV_LEVEL_5_SYM		(NV_LEVEL_5|NV_LEVEL_5_SYM_FLAG)
+
+	lba_t    stripeBlockSize;	/* Array stripe block size in sectors */
+	uint32_t stripeBlockByteSize;	/* stripeBlockSize in bytes */
+	uint32_t stripeBlockPower;	/* Array stripe block size in log2 */
+	lba_t    stripeMask;	/* stripeBlockSize - 1 */
+	lba_t    stripeSize;	/* stripeBlockSize * stripeWidth */
+	uint32_t stripeByteSize;	/* stripeSize in bytes */
+	lba_t    raidJobMark;	/* Ignored if array is idle, otherwise the */
+                           	/* LBA where job is finished               */
+	uint32_t originalLevel;	/* RAID level before morph */
+	lba_t    originalCapacity;	/* Array capacity before morph */
+
+	uint32_t flags;		/* Flags for array */
+#define NV_ARRAY_FLAG_BOOT		(0x00000001) /* BIOS use only */
+#define NV_ARRAY_FLAG_ERROR		(0x00000002) /* Degraded or offling */
+#define NV_ARRAY_FLAG_PARITY_VALID	(0x00000004) /* RAID-3/5 parity valid */
+#define	NV_BROKEN(n)		(n->array.flags & NV_ARRAY_FLAG_ERROR)
+#define	NV_SET_BROKEN(n)	(n->array.flags |= NV_ARRAY_FLAG_ERROR)
+} __attribute__ ((packed));
+
+/* Ondisk metadata */
+struct nv {
+	uint8_t  vendor[NV_ID_LENGTH]; /* 0x00 - 0x07 ID string */
+	uint32_t size;		/* 0x08 - 0x0B Size of metadata in dwords */
+	uint32_t chksum;	/* 0x0C - 0x0F Checksum of this struct */
+	uint16_t version;	/* 0x10 - 0x11 NV version */
+	uint8_t  unitNumber;	/* 0x12 Disk index in array */
+	uint8_t  reserved;	/* 0x13 */
+	lba_t    capacity;	/* 0x14 - 0x17 Array capacity in sectors */ 
+	uint32_t sectorSize;	/* 0x18 - 0x1B Sector size */
+#define	NV_PRODUCTIDS	16
+				/* 0x1C - 0x2B Array product ID */
+	uint8_t  productID[NV_PRODUCTIDS];
+				/*             Match INQUIRY data */
+#define	NV_PRODUCTREVISIONS	4
+				/* 0x2C - 0x2F Array product revision */
+	uint8_t  productRevision[NV_PRODUCTREVISIONS];
+				/*             Match INQUIRY data */
+	uint32_t unitFlags;	/* 0x30 - 0x33 Flags for this disk */
+	struct nv_array_base array;      /* Array information */
+} __attribute__ ((packed));
+#endif
+
+/* Prototype of the register function for this metadata format handler */
+int register_nv(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/pdc.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/pdc.c
+++ -	2008-02-22 16:57:42.201893000 +0000
@@ -0,0 +1,478 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Promise FastTrak ATARAID metadata format handler.
+ *
+ * pdc_read() and pdc_group() profited from
+ * Carl-Daniel Hailfinger's raiddetect code.
+ */
+#define	HANDLER	"pdc"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "pdc.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/*
+ * Make up Promise RAID device name.
+ */
+static unsigned int set_number(struct pdc *pdc)
+{
+	return pdc->raid.disk_number >= (pdc->raid.total_disks / 2);
+}
+
+static size_t _name(struct pdc *pdc, char *str, size_t len, int subset)
+{
+	return snprintf(str, len, subset ? "pdc_%u-%u" : "pdc_%u",
+			pdc->raid.magic_1, set_number(pdc));
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	struct pdc *pdc = META(rd, pdc);
+	size_t len;
+	char *ret = NULL;
+
+	if ((ret = dbg_malloc((len = _name(pdc, ret, 0, subset) + 1)))) {
+		_name(pdc, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN,
+			 len - HANDLER_LEN - (subset ? 2 : 0));
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/*
+ * Retrieve status of device.
+ *
+ * FIXME: need to identify state definitions.
+ */
+static enum status status(struct pdc *pdc)
+{
+	return PDC_BROKEN(pdc) ? s_broken : s_ok;
+}
+
+/*
+ * Mapping of Promise types to generic types.
+ */
+#define	PDC_T_RAID10	0x2	/* Not defind by Promise (yet). */
+static struct types types[] = {
+        { PDC_T_SPAN,   t_linear},
+        { PDC_T_RAID0,  t_raid0},
+        { PDC_T_RAID1,  t_raid1},
+        { PDC_T_RAID10, t_raid0},
+        { 0, t_undef}
+};
+
+static int is_raid10(struct pdc *pdc)
+{
+	return pdc->raid.type == PDC_T_RAID10 ||
+	       (pdc->raid.type == PDC_T_RAID1 && pdc->raid.total_disks > 3);
+}
+
+/* Neutralize disk type */
+static enum type type(struct pdc *pdc)
+{
+	if (is_raid10(pdc))
+		pdc->raid.type = PDC_T_RAID10;
+
+	return rd_type(types, (unsigned int) pdc->raid.type);
+}
+
+/* Calculate checksum on Promise metadata. */
+static uint32_t checksum(struct pdc *pdc)
+{
+	unsigned int i = 511, sum = 0;
+	uint32_t *p = (uint32_t*) pdc;
+
+	while (i--)
+		sum += *p++;
+
+	return sum == pdc->checksum;
+}
+
+/* Calculate metadata offset. */
+/*
+ * Read a Promise FastTrak RAID device
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct pdc *pdc = meta;
+	struct pdc_disk *disk;
+
+	CVT32(pdc->magic_0);
+	CVT32(pdc->magic_1);
+	CVT32(pdc->raid.flags);
+	CVT32(pdc->raid.magic_0);
+	CVT32(pdc->raid.disk_secs);
+	CVT32(pdc->raid.total_secs);
+	CVT16(pdc->raid.cylinders);
+	CVT32(pdc->raid.magic_1);
+
+	for (disk = pdc->raid.disk;
+	     disk < pdc->raid.disk + pdc->raid.total_disks;
+	     disk++) {
+		CVT32(disk->magic_0);
+		CVT32(disk->disk_number);
+	}
+}
+#endif
+
+/* Read and try to discover Promise signature. */
+static void *pdc_read_metadata(struct lib_context *lc, struct dev_info *di,
+			       size_t *size, uint64_t *offset,
+			       union read_info *info)
+{
+	/* Assume certain sectors off the end of the RAID device. */
+	static unsigned int *s, sectors[] = {
+		PDC_CONFIGOFFSETS, 0,
+	};
+	struct pdc *ret;
+
+	if ((ret = alloc_private(lc, handler, sizeof(*ret)))) {
+		for (s = sectors; *s; s++) {
+			if (read_file(lc, handler, di->path, ret, sizeof(*ret),
+				      (di->sectors - *s) << 9) &&
+			    !strncmp((const char*) ret->promise_id, PDC_MAGIC,
+				     PDC_ID_LENGTH)) {
+				info->u32 = *s;
+				return (void*) ret;
+			}
+		}
+
+		dbg_free(ret);
+	}
+
+	return NULL;
+}
+
+/* Magic check. */
+static int is_pdc(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct pdc *pdc = meta;
+
+	/*
+	 * No we've got the PDC magic string veryfied, we can
+	 * check that the rest of the metadata is valid.
+	 */
+	if (pdc->magic_0 == pdc->raid.magic_0 &&
+	    pdc->raid.total_disks &&
+	    pdc->raid.total_disks < PDC_MAXDISKS)
+		return 1;
+
+
+	LOG_ERR(lc, 0, "%s: identifying %s, magic_0: 0x%x/0x%x, "
+		"magic_1: 0x%x/0x%x, total_disks: %u",
+		handler, di->path,
+		pdc->magic_0, pdc->raid.magic_0,
+		pdc->magic_1, pdc->raid.magic_1, pdc->raid.total_disks);
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *pdc_read(struct lib_context *lc, struct dev_info *di)
+{ 
+	return read_raid_dev(lc, di, pdc_read_metadata, 0, 0, to_cpu, is_pdc,
+			     NULL, setup_rd, handler);
+}
+
+/*
+ * Write a Promise FastTrak RAID device.
+ */
+static int pdc_write(struct lib_context *lc, struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct pdc *pdc = META(rd, pdc);
+
+	to_disk(pdc);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(pdc);
+#endif
+	return ret;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD(new), pdc))->raid.disk_number <
+	       (META(RD(pos), pdc))->raid.disk_number;
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return !set_number(META(RD_RS(RS(new)), pdc));
+}
+
+/*
+ * Group the RAID disk into a Promise set.
+ */
+static unsigned int stride(struct pdc *pdc)
+{
+	return pdc->raid.raid0_shift ? 1 << pdc->raid.raid0_shift : 0;
+}
+
+static void super_created(struct raid_set *super, void *private)
+{
+	super->type   = t_raid1;
+	super->stride = stride(META((private), pdc));
+}
+
+/* Add a PDC RAID device to a set. */
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct pdc *pdc = META(rd, pdc);
+
+	if (!init_raid_set(lc, rs, rd, stride(pdc), pdc->raid.type, handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (pdc->raid.type) {
+	case PDC_T_SPAN:
+	case PDC_T_RAID0:
+	case PDC_T_RAID1:
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case PDC_T_RAID10:
+		/*
+		 * RAID10:
+		 *
+		 * We've got a striped raid set with a mirror on top
+		 * when we get here.
+		 * Let's find and optionally allocate the mirror set on top.
+		 */
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+static struct raid_set *pdc_group(struct lib_context *lc, struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/*
+ * Check a PDC RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	struct pdc *pdc = META(rd, pdc);
+
+	if (context && pdc->raid.type != PDC_T_SPAN)
+		*((uint64_t*) context) += rd->sectors;
+
+	return pdc->raid.total_disks;
+}
+
+static int check_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_dev *rd, void *context)
+{
+      return *((uint64_t*) context) >= (META(rd, pdc))->raid.total_secs;
+}
+
+static int pdc_check(struct lib_context *lc, struct raid_set *rs)
+{
+	uint64_t total_secs = 0;
+
+	/*
+	 * Calculate total_secs in 1st check_raid_set() run and
+	 * check the device sizes against that in the 2nd run.
+	 */
+	return check_raid_set(lc, rs, devices, &total_secs,
+			      NO_CHECK_RD, NULL, handler) &&
+	       check_raid_set(lc, rs, devices, NULL,
+			      check_rd, &total_secs, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct pdc *pdc = META(rd, pdc);
+
+	/* Avoid write trashing. */
+	if (status(pdc) & s_broken)
+		return 0;
+
+	PDC_SET_BROKEN(pdc);
+
+	return 1;
+}
+
+static struct event_handlers pdc_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/* Log native information about a Promise RAID device. */
+static void pdc_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int i;
+	struct pdc *pdc = META(rd, pdc);
+	struct pdc_disk *disk;
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	DP("promise_id: \"%s\"", pdc, pdc->promise_id);
+	DP("unknown_0: 0x%x", pdc, pdc->unknown_0);
+	DP("magic_0: 0x%x", pdc, pdc->magic_0);
+	DP("unknown_1: 0x%x", pdc, pdc->unknown_1);
+	DP("magic_1: 0x%x", pdc, pdc->magic_1);
+	DP("unknown_2: 0x%x", pdc, pdc->unknown_2);
+	DP("raid.flags: 0x%x", pdc, pdc->raid.flags);
+	DP("raid.unknown_0: 0x%x", pdc, pdc->raid.unknown_0);
+	DP("raid.disk_number: %u", pdc, pdc->raid.disk_number);
+	DP("raid.channel: %u", pdc, pdc->raid.channel);
+	DP("raid.device: %u", pdc, pdc->raid.device);
+	DP("raid.magic_0: 0x%x", pdc, pdc->raid.magic_0);
+	DP("raid.unknown_1: 0x%x", pdc, pdc->raid.unknown_1);
+	DP("raid.unknown_2: 0x%x", pdc, pdc->raid.unknown_2);
+	DP("raid.disk_secs: %u", pdc, pdc->raid.disk_secs);
+	DP("raid.unknown_3: 0x%x", pdc, pdc->raid.unknown_3);
+	DP("raid.unknown_4: 0x%x", pdc, pdc->raid.unknown_4);
+	DP("raid.status: 0x%x", pdc, pdc->raid.status);
+	DP("raid.type: 0x%x", pdc, pdc->raid.type);
+	DP("raid.total_disks: %u", pdc, pdc->raid.total_disks);
+	DP("raid.raid0_shift: %u", pdc, pdc->raid.raid0_shift);
+	DP("raid.raid0_disks: %u", pdc, pdc->raid.raid0_disks);
+	DP("raid.array_number: %u", pdc, pdc->raid.array_number);
+	DP("raid.total_secs: %u", pdc, pdc->raid.total_secs);
+	DP("raid.cylinders: %u", pdc, pdc->raid.cylinders);
+	DP("raid.heads: %u", pdc, pdc->raid.heads);
+	DP("raid.sectors: %u", pdc, pdc->raid.sectors);
+	DP("raid.magic_1: 0x%x", pdc, pdc->raid.magic_1);
+	DP("raid.unknown_5: 0x%x", pdc, pdc->raid.unknown_5);
+
+	for (disk = pdc->raid.disk, i = 0;
+	     i < pdc->raid.total_disks;
+	     disk++, i++) {
+		P2("raid.disk[%d].unknown_0: 0x%x", pdc, i, disk->unknown_0);
+		P2("raid.disk[%d].channel: %u", pdc, i, disk->channel);
+		P2("raid.disk[%d].device: %u", pdc, i, disk->device);
+		P2("raid.disk[%d].magic_0: 0x%x", pdc, i, disk->magic_0);
+		P2("raid.disk[%d].disk_number: %u", pdc, i, disk->disk_number);
+	}
+
+	P("checksum: 0x%x %s", pdc, pdc->checksum, pdc->checksum,
+	   checksum(pdc) ? "Ok" : "BAD");
+}
+#endif
+
+static struct dmraid_format pdc_format = {
+	.name	= HANDLER,
+	.descr	= "Promise FastTrack",
+	.caps	= "S,0,1,10",
+	.format = FMT_RAID,
+	.read	= pdc_read,
+	.write	= pdc_write,
+	.group	= pdc_group,
+	.check	= pdc_check,
+	.events	= &pdc_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= pdc_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_pdc(struct lib_context *lc)
+{
+	return register_format_handler(lc, &pdc_format);
+}
+
+/* Calculate RAID device size in sectors depending on RAID type. */
+static uint64_t sectors(struct raid_dev *rd, unsigned int meta_sector)
+{
+	struct pdc *pdc = META(rd, pdc);
+
+	switch (pdc->raid.type) {
+	case PDC_T_RAID10:
+               return pdc->raid.total_secs / (pdc->raid.total_disks / 2);
+
+	case PDC_T_RAID1:
+               return pdc->raid.total_secs;
+
+	case PDC_T_RAID0:
+               return pdc->raid.total_secs / pdc->raid.total_disks;
+
+	case PDC_T_SPAN:
+	       return rd->di->sectors - meta_sector;
+	}
+
+	return 0;
+}
+
+/* Set the RAID device contents up derived from the PDC ones */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	unsigned int meta_sector;
+	struct pdc *pdc = meta;
+
+	if (!checksum(pdc))
+		LOG_ERR(lc, 0, "%s: invalid checksum on %s", handler, di->path);
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	meta_sector = info->u32;
+	rd->meta_areas->offset = di->sectors - meta_sector;
+	rd->meta_areas->size = sizeof(*pdc);
+	rd->meta_areas->area = (void*) pdc;
+
+        rd->di = di;
+	rd->fmt = &pdc_format;
+
+	rd->status = status(pdc);
+	/*
+	 * Type needs to be set before sectors(), because we need
+	 * to set the RAID10 type used there!
+	 */
+	rd->type   = type(pdc);
+
+	rd->offset = PDC_DATAOFFSET;
+	if (!(rd->sectors = sectors(rd, meta_sector)))
+		return log_zero_sectors(lc, di->path, handler);
+
+        return (rd->name = name(lc, rd, is_raid10(pdc))) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/pdc.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/pdc.h
+++ -	2008-02-22 16:57:42.284835000 +0000
@@ -0,0 +1,81 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_PDC_H_
+#define	_PDC_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+#define PDC_CONFIGOFFSETS	63,255,256,16,399,735
+#define	PDC_DATAOFFSET 0
+
+/* Ondisk metadata for Promise Fastrack */
+struct pdc {
+#define PDC_ID_LENGTH	24
+	uint8_t promise_id[PDC_ID_LENGTH];	/* 0x00 - 0x17 */
+#define PDC_MAGIC        "Promise Technology, Inc."
+
+	uint32_t unknown_0;			/* 0x18 - 0x1B */
+	uint32_t magic_0;			/* 0x1C - 0x1F */
+	uint32_t unknown_1;			/* 0x20 - 0x23 */
+	uint32_t magic_1;			/* 0x24 - 0x27 */
+	uint16_t unknown_2;			/* 0x28 - 0x2B */
+	uint8_t filler1[470];			/* 0x2C - 0x1FF */
+	struct {
+		uint32_t flags;			/* 0x200 - 0x203 */
+		uint8_t unknown_0;		/* 0x204 */
+		uint8_t disk_number;		/* 0x205 */
+		uint8_t channel;		/* 0x206 */
+		uint8_t device;			/* 0x207 */
+		uint32_t magic_0;		/* 0x208 - 0x20B */
+		uint32_t unknown_1;		/* 0x20C - 0x20F */
+		uint32_t unknown_2;		/* 0x210 - 0x213 */
+		uint32_t disk_secs;		/* 0x214 - 0x217 */
+		uint32_t unknown_3;		/* 0x218 - 0x21B */
+		uint16_t unknown_4;		/* 0x21C - 0x21D */
+		uint8_t status;			/* 0x21E */
+/* FIXME: bit 0x80 doesn't seem to indicate error as previously assumed. */
+// #define	PDC_BROKEN(pdc)		((pdc)->raid.status &  0x80)
+#define	PDC_BROKEN(pdc)		((pdc)->raid.status &  0x00)
+#define	PDC_SET_BROKEN(pdc)	((pdc)->raid.status |= 0x80)
+		uint8_t type;			/* 0x21F */
+#define	PDC_T_RAID0	0x0
+#define	PDC_T_RAID1	0x1
+#define	PDC_T_SPAN	0x8
+		uint8_t total_disks;		/* 0x220 */
+		uint8_t raid0_shift;		/* 0x221 */
+		uint8_t raid0_disks;		/* 0x222 */
+		uint8_t array_number;		/* 0x223 */
+		uint32_t total_secs;		/* 0x224 - 0x227 */
+		uint16_t cylinders;		/* 0x228 - 0x229 */
+		uint8_t heads;			/* 0x22A */
+		uint8_t sectors;		/* 0x22B */
+		uint32_t magic_1;		/* 0x22C - 0x2EF */
+		uint32_t unknown_5;		/* 0x230 - 0x233 */
+		struct pdc_disk {
+			uint16_t unknown_0;	/* 0x234 - 0x235 */
+			uint8_t channel;	/* 0x236 */
+			uint8_t device;		/* 0x237 */
+			uint32_t magic_0;	/* 0x238 - 0x23B */
+			uint32_t disk_number;	/* 0x23C - 0x23F */
+		} disk[8];
+	} raid;
+
+	uint32_t filler2[346];			/* 0x294 - */
+	uint32_t checksum;
+} __attribute__ ((packed));
+
+#define	PDC_MAXDISKS	8
+
+#endif
+
+int register_pdc(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/sil.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/sil.c
+++ -	2008-02-22 16:57:42.375226000 +0000
@@ -0,0 +1,602 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Silicon Image Medley ATARAID metadata format handler.
+ */
+#define	HANDLER	"sil"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "sil.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/* Make up RAID device name from some 'magic' numbers */
+/* FIXME: better name ? */
+static size_t _name(struct sil *sil, char *str, size_t len, unsigned int subset)
+{
+	return snprintf(str, len,
+			subset ? "sil_%02u%02u%02u%02u%02u%02u-%u" :
+				 "sil_%02u%02u%02u%02u%02u%02u",
+			sil->year, sil->month, sil->day,
+			sil->hour, sil->minutes % 60, sil->seconds % 60,
+			sil->type == SIL_T_RAID1 ? sil->mirrored_set_number :
+						   sil->striped_set_number);
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret;
+	struct sil *sil = META(rd, sil);
+
+	subset = subset && sil->type == SIL_T_RAID10;
+	if ((ret = dbg_malloc((len = _name(sil, NULL, 0, subset) + 1)))) {
+		_name(sil, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN -
+			 (strrchr(ret, '-') ? 3 : 1));
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/*
+ * Retrieve status of device.
+ * FIXME: is this sufficient to cover all state ?
+ */
+static enum status status(struct sil *sil)
+{
+	switch (sil->mirrored_set_state) {
+	case SIL_OK:
+	case SIL_MIRROR_SYNC:
+		return s_ok;
+
+	case SIL_MIRROR_NOSYNC:
+		return s_nosync;
+	}
+
+	return s_broken;
+}
+
+/* Mapping of SIL 680 types to generic types */
+static struct types types[] = {
+        { SIL_T_SPARE,  t_spare},
+        { SIL_T_RAID0,  t_raid0},
+        { SIL_T_RAID5, 	t_raid5_ls},
+        { SIL_T_RAID1,  t_raid1},
+        { SIL_T_RAID10, t_raid0},
+        { 0,            t_undef}
+};
+
+/* Neutralize disk type */
+static enum type type(struct sil *sil)
+{
+	return rd_type(types, (unsigned int) sil->type);
+}
+
+/* Calculate checksum on metadata */
+static int checksum(struct sil *sil)
+{
+	int sum = 0;
+	unsigned int count = struct_offset(sil, checksum1) / 2;
+	uint16_t *p = (uint16_t*) sil;
+
+        while (count--)
+		sum += *p++;
+
+	return (-sum & 0xFFFF) == sil->checksum1;
+}
+
+/*
+ * Read a Silicon Image RAID device
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct sil *sil = meta;
+
+	CVT32(sil->magic);
+	CVT32(sil->array_sectors_low);
+	CVT32(sil->array_sectors_high);
+	CVT32(sil->thisdisk_sectors);
+	CVT16(sil->product_id);
+	CVT16(sil->vendor_id);
+	CVT16(sil->minor_ver);
+	CVT16(sil->major_ver);
+	CVT16(sil->raid0_stride);
+	CVT32(sil->rebuild_ptr_low);
+	CVT32(sil->rebuild_ptr_high);
+	CVT32(sil->incarnation_no);
+	CVT16(sil->checksum1);
+	CVT16(sil->checksum2);
+}
+#endif
+
+#define	AREAS	4
+#define SIL_META_AREA(i)	(SIL_CONFIGOFFSET - (i * 512 << 9))
+
+static inline int is_sil(struct sil *sil)
+{
+	return SIL_MAGIC_OK(sil) && sil->disk_number < 8;
+}
+
+static int sil_valid(struct lib_context *lc, struct dev_info *di,
+		     void *meta, unsigned int area)
+{
+	struct sil *sil = meta;
+
+	if (!is_sil(sil))
+		return 0;
+
+	if (sil->major_ver != 2)
+		log_warn(lc, "%s: major version %u in area %u; format handler "
+			 "tested on version 2 only",
+			 handler, sil->major_ver, area);
+
+	if (!checksum(sil))
+		LOG_ERR(lc, 0, "%s: invalid metadata checksum in area %u on %s",
+			handler, area, di->path);
+
+	if (di->sectors < sil->thisdisk_sectors)
+		LOG_ERR(lc, 0,
+			"%s: invalid disk size in metadata area %u on %s",
+			handler, area, di->path);
+
+	return 1;
+}
+
+static void free_sils(struct sil **sils, unsigned int i)
+{
+	for (; i < AREAS; i++)
+		dbg_free(sils[i]);
+
+	dbg_free(sils);
+}
+
+static void *sil_read_metadata(struct lib_context *lc, struct dev_info *di,
+			       size_t *size, uint64_t *offset,
+			       union read_info *info)
+{
+	unsigned int i, valid;
+	char str[9] = { 0, };
+	struct sil *sil, **sils;
+
+	if (!(sils = dbg_malloc(AREAS * sizeof(*sils))))
+		goto out;
+
+	memset(sils, 0, AREAS * sizeof(*sils));
+
+	/* Read the 4 metadata areas. */
+	for (i = valid = 0; i < AREAS; i++) {
+		if (!(sil = alloc_private_and_read(lc, handler, sizeof(*sil),
+						   di->path, SIL_META_AREA(i))))
+			goto bad;
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+		to_cpu(sil);
+#endif
+		/* Create string with list of valid areas. */
+		if (sil_valid(lc, di, sil, i + 1)) {
+			sils[valid] = sil;
+			sprintf(&str[strlen(str)], "%s%u",
+			        valid++ ? "," : "", i + 1);
+		} else
+			dbg_free(sil);
+	}
+
+	if (valid) {
+		log_notice(lc, "%s: area%s %s[%u] %s valid",
+			   handler, valid ? "s" : "", str, AREAS,
+			   valid == 1 ? "is" : "are");
+		goto out;
+	}
+	
+   bad:
+	free_sils(sils, 0);
+	sils = NULL;
+
+   out:
+	return (void*) sils;
+}
+
+static int _file_name(char *str, size_t len, char *n, int i)
+{
+	return snprintf(str, len, "%s_%d", n, i) + 1;
+}
+
+static char *file_name(struct lib_context *lc, char *n, int i)
+{
+	size_t len;
+	char *ret;
+
+	if ((ret = dbg_malloc((len = _file_name(NULL, 0, n, i)))))
+		_file_name(ret, len, n, i);
+	else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/* File all metadata areas. */
+static void sil_file_metadata(struct lib_context *lc, struct dev_info *di,
+			      void *meta)
+{
+	unsigned int i;
+	char *n;
+	struct sil **sils = meta;
+	
+	for (i = 0; i < AREAS; i++) {
+		if (!(n = file_name(lc, di->path, i)))
+			break;
+
+		file_metadata(lc, handler, n, sils[i],
+			      sizeof(**sils), SIL_META_AREA(i));
+		dbg_free(n);
+	}
+
+	file_dev_size(lc, handler, di);
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *sil_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, sil_read_metadata, 0, 0, NULL, NULL,
+			     sil_file_metadata, setup_rd, handler);
+
+}
+
+/*
+ * Write a Silicon Image RAID device.
+ */
+static int sil_write(struct lib_context *lc, struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct sil *sil = META(rd, sil);
+
+	to_disk(sil);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(sil);
+#endif
+	return ret;
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD(new), sil))->disk_number <
+	       (META(RD(pos), sil))->disk_number;
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD_RS(RS(new)), sil))->mirrored_set_number <
+	       (META(RD_RS(RS(pos)), sil))->mirrored_set_number;
+}
+
+/*
+ * Group the RAID disk into a SIL set.
+ *
+ * Check device hierarchy and create super set appropriately.
+ */
+static void super_created(struct raid_set *ss, void *private)
+{
+	ss->type   = t_raid1;
+	ss->stride = META(private, sil)->raid0_stride;
+}
+
+/* FIXME: handle spares. */
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct sil *sil = META(rd, sil);
+
+	if (!init_raid_set(lc, rs, rd, sil->raid0_stride, sil->type, handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (sil->type) {
+	case SIL_T_RAID0:
+	case SIL_T_RAID1:
+	case SIL_T_RAID5:
+		if (!(find_set(lc, NULL, rs->name, FIND_TOP)))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case SIL_T_RAID10:
+		/*
+		 * We've got a striped raid set with a mirror set on top
+		 * when we get here.
+		 * Let's find and optionally allocate the mirror set.
+		 */
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+/* Add a SIL RAID device to a set */
+static struct raid_set *sil_group(struct lib_context *lc, struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+/*
+ * Check a SIL RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	int ret;
+	struct sil *sil = META(rd, sil);
+
+	switch (sil->type) {
+	case SIL_T_RAID0:
+	case SIL_T_RAID10:
+		ret = sil->drives_per_striped_set;
+		break;
+
+	case SIL_T_RAID1:
+		ret = sil->drives_per_mirrored_set;
+		break;
+
+	default:
+		ret = 0;
+	}
+
+	return ret;
+}
+
+static int sil_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL,
+			      NO_CHECK_RD, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct sil *sil = META(rd, sil);
+
+	/* Avoid write trashing. */
+	if (status(sil) & s_broken)
+		return 0;
+
+	sil->member_status = 0;
+
+	return 1;
+}
+
+static struct event_handlers sil_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about a Silicon Image  RAID device.
+ */
+static void sil_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	char *tt;
+	struct sil *sil = META(rd, sil);
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	DP("unknown0: \"%42s\"", sil, sil->unknown0);
+	DP("ascii_version: \"%8s\"", sil, sil->ascii_version);
+	DP("diskname: \"%32s\"", sil, sil->diskname);
+	DP("unknown1: \"%22s\"", sil, sil->unknown1);
+	DP("magic: 0x%x", sil, sil->magic);
+	DP("unknown1a: \"%8s\"", sil, sil->unknown1a);
+	DP("array_sectors_low: %u", sil, sil->array_sectors_low);
+	DP("array_sectors_high: %u", sil, sil->array_sectors_high);
+	DP("unknown2: \"%4s\"", sil, sil->unknown2);
+	DP("thisdisk_sectors: %u", sil, sil->thisdisk_sectors);
+	DP("product_id: %u", sil, sil->product_id);
+	DP("vendor_id: %u", sil, sil->vendor_id);
+	DP("minor_ver: %u", sil, sil->minor_ver);
+	DP("major_ver: %u", sil, sil->major_ver);
+	DP("seconds: %u", sil, sil->seconds % 60);
+	DP("seconds(full): 0x%x", sil, sil->seconds);
+	DP("minutes: %u", sil, sil->minutes % 60);
+	DP("minutes(full): 0x%x", sil, sil->minutes);
+	DP("hour: %u", sil, sil->hour);
+	DP("day: %u", sil, sil->day);
+	DP("month: %u", sil, sil->month);
+	DP("year: %u", sil, sil->year);
+	DP("raid0_stride: %u", sil, sil->raid0_stride);
+	DP("disk_number: %u", sil, sil->disk_number);
+	DP("type: %u", sil, sil->type);
+	DP("drives_per_striped_set: %d", sil, sil->drives_per_striped_set);
+	DP("striped_set_number: %d", sil, sil->striped_set_number);
+	DP("drives_per_mirrored_set: %d", sil, sil->drives_per_mirrored_set);
+	DP("mirrored_set_number: %d", sil, sil->mirrored_set_number);
+	DP("rebuild_ptr_low: %u", sil, sil->rebuild_ptr_low);
+	DP("rebuild_ptr_high: %u", sil, sil->rebuild_ptr_high);
+	DP("incarnation_no: %u", sil, sil->incarnation_no);
+	DP("member_status: %u", sil, sil->member_status);
+	DP("mirrored_set_state: %u", sil, sil->mirrored_set_state);
+	DP("reported_device_location: %u", sil, sil->reported_device_location);
+	DP("idechannel: %u", sil, sil->idechannel);
+	DP("auto_rebuild: %u", sil, sil->auto_rebuild);
+
+	if ((tt = dbg_strndup(sil->text_type, 16))) {
+		P("text_type: \"%s\"", sil, sil->text_type, tt);
+		dbg_free(tt);
+	}
+
+	DP("checksum1: %u", sil, sil->checksum1);
+	DP("checksum2: %u", sil, sil->checksum2);
+}
+#endif
+
+static struct dmraid_format sil_format = {
+	.name	= HANDLER,
+	.descr	= "Silicon Image(tm) Medley(tm)",
+	.caps	= "0,1,10",
+	.format = FMT_RAID,
+	.read	= sil_read,
+	.write	= sil_write,
+	.group	= sil_group,
+	.check	= sil_check,
+	.events	= &sil_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= sil_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_sil(struct lib_context *lc)
+{
+	return register_format_handler(lc, &sil_format);
+}
+
+/* Set the RAID device contents up derived from the SIL ones. */
+static int stripes(struct sil *sil)
+{
+	return sil->drives_per_striped_set > -1 &&
+	       sil->disk_number < sil->drives_per_striped_set;
+}
+
+static uint64_t sectors(struct raid_dev *rd)
+{
+	uint64_t array_sectors, ret = 0;
+	struct sil *sil = META(rd, sil);
+
+	array_sectors = (((uint64_t) sil->array_sectors_high) << 32) +
+			 sil->array_sectors_low;
+
+	switch (sil->type) {
+	case SIL_T_SPARE:
+		/* Cook them up... */
+		ret = rd->di->sectors - (AREAS - 1) * 512 -
+		      ((rd->di->sectors & 1) ? 1 : 2);
+		break;
+
+	case SIL_T_RAID0:
+		if (stripes(sil))
+			ret = array_sectors / sil->drives_per_striped_set;
+		break;
+
+	case SIL_T_RAID1:
+	case SIL_T_RAID10:
+		ret = array_sectors;
+		break;
+
+	default:
+		/* Cook them up... */
+		ret = rd->di->sectors - (AREAS - 1) * 512 -
+		      ((rd->di->sectors & 1) ? 1 : 2);
+		break;
+	}
+
+	return ret;
+}
+
+/* Quorate SIL metadata copies. */
+static struct sil *quorate(struct lib_context *lc, struct dev_info *di,
+			   struct sil *sils[])
+{
+	unsigned int areas = 0, i, j;
+	struct sil *sil;
+	
+	/* Count valid metadata areas. */
+	while (areas < AREAS && sils[areas++]);
+
+	if (areas == AREAS)
+		goto out;
+
+	log_err(lc, "%s: only %u/%u metadata areas found on %s, %sing...",
+		handler, areas, AREAS, di->path, areas > 1 ? "elect" : "pick");
+
+	/* Identify maximum identical copies. */
+	for (i = 0; i < areas; i++) {
+		for (j = i + 1, sil = sils[i]; j < areas; j++) {
+			if (!memcmp(sil, sils[j], sizeof(*sil)))
+				goto end;
+		}
+	}
+
+   end:
+	if (i) {
+		sil = sils[0];
+		sils[0] = sils[i];
+		sils[i] = sil;
+	}
+
+   out:
+	return sils[0];
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	unsigned int i;
+	struct meta_areas *ma;
+	struct sil *sil, **sils = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, AREAS)))
+		goto bad;
+
+	sil = quorate(lc, di, sils);	/* Quorate one copy+save a pointer.*/
+	free_sils(sils, 1);		/* Free the other copies. */
+
+	for (i = 0, ma = rd->meta_areas; i < rd->areas; i++, ma++) {
+		ma->offset = SIL_META_AREA(i) >> 9;
+		ma->size = sizeof(*sil);
+		ma->area = (void*) sil;
+	}
+
+	rd->di = di;
+	rd->fmt = &sil_format;
+
+	rd->offset = SIL_DATAOFFSET;
+	if (!(rd->sectors = sectors(rd)))
+		return log_zero_sectors(lc, di->path, handler);
+
+	rd->status = status(sil);
+	rd->type   = type(sil);
+
+        return (rd->name = name(lc, rd, sil->type == SIL_T_RAID10)) ? 1 : 0;
+
+   bad:
+	free_sils(sils, 0);
+
+	return 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/sil.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/sil.h
+++ -	2008-02-22 16:57:42.455990000 +0000
@@ -0,0 +1,79 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_SIL_H_
+#define	_SIL_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+#define SIL_CONFIGOFFSET        ((di->sectors - 1) << 9)
+#define SIL_DATAOFFSET          0       /* Data offset in sectors */
+
+struct sil {
+        uint8_t	unknown0[0x2E];		/* 0x4 - 0x2D */
+        uint8_t	ascii_version[0x36 - 0x2E];/* 0x2E - 0x35 */
+	int8_t		diskname[0x56 - 0x36];	/* 0x36 - 0x55 */
+	int8_t		unknown1[0x60 - 0x56];	/* 0x56 - 0x59 */	
+	uint32_t	magic;			/* 0x60 - 0x63 */
+#define	SIL_MAGIC	0x3000000
+#define	SIL_MAGIC_OK(sil)	((sil->magic & 0x3ffffff) == SIL_MAGIC)
+	int8_t		unknown1a[0x6C - 0x64];	/* 0x64 - 0x6B */	
+	uint32_t	array_sectors_low;	/* 0x6C - 0x6F */
+	uint32_t	array_sectors_high;	/* 0x70 - 0x73 */
+	int8_t		unknown2[0x78 - 0x74];	/* 0x74 - 0x77 */
+	uint32_t	thisdisk_sectors;	/* 0x78 - 0x7B */
+	int8_t		unknown3[0x100 - 0x7C];	/* 0x7C - 0xFF */
+	int8_t		unknown4[0x104 - 0x100];/* 0x100 - 0x103 */
+	uint16_t	product_id;		/* 0x104 + 0x105 */
+	uint16_t	vendor_id;		/* 0x106 + 0x107 */
+        uint16_t	minor_ver;		/* 0x108 + 0x109 */
+        uint16_t	major_ver;		/* 0x10A + 0x10B */
+	uint8_t	seconds;		/* 0x10C */
+	uint8_t	minutes;		/* 0x10D */
+	uint8_t	hour;			/* 0x10E */
+	uint8_t	day;			/* 0x10F */
+	uint8_t	month;			/* 0x110 */
+	uint8_t	year;			/* 0x111 */
+	uint16_t	raid0_stride;		/* 0x112 + 0x113 */
+	int8_t		unknown6[0x116 - 0x114];/* 0x114 + 0x115 */
+	uint8_t	disk_number;		/* 0x116 */
+	uint8_t	type;			/* 0x117 */
+#define	SIL_T_RAID0	0
+#define	SIL_T_RAID1	1
+#define	SIL_T_RAID10	2
+#define	SIL_T_RAID5	16
+#define	SIL_T_SPARE	3
+	int8_t		drives_per_striped_set;	/* 0x118 */
+	int8_t		striped_set_number;	/* 0x119 */
+        int8_t		drives_per_mirrored_set;/* 0x11A */
+        int8_t		mirrored_set_number;	/* 0x11B */
+	uint32_t	rebuild_ptr_low;	/* 0x11C - 0x12F */
+	uint32_t	rebuild_ptr_high;	/* 0x120 - 0x123 */
+	uint32_t	incarnation_no;		/* 0x124 - 0x127 */
+        uint8_t	member_status;		/* 0x128 */
+        uint8_t	mirrored_set_state;	/* 0x129 */
+#define	SIL_OK			0
+#define	SIL_MIRROR_NOSYNC	1
+#define	SIL_MIRROR_SYNC		2
+        uint8_t	reported_device_location;/* 0x12A */
+        uint8_t	idechannel;		/* 0x12B */
+        uint8_t	auto_rebuild;		/* 0x12C */
+#define	SIL_MIRROR_NOAUTOREBUILD	0
+	uint8_t		unknown8;		/* 0x12D */
+	uint8_t		text_type[0x13E - 0x12E]; /* 0x12E - 0x13D */
+	uint16_t	checksum1;		/* 0x13E + 0x13F */	
+	int8_t		assumed_zeros[0x1FE - 0x140];/* 0x140 - 0x1FD */
+	uint16_t	checksum2;		/* 0x1FE + 0x1FF */
+} __attribute__ ((packed));
+#endif
+
+int register_sil(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/ataraid/via.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/via.c
+++ -	2008-02-22 16:57:42.538899000 +0000
@@ -0,0 +1,413 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file DISCLAIMER at the top of this source tree for license information.
+ */
+
+/*
+ * VIA metadata format handler.
+ */
+#define	HANDLER	"via"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "via.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+
+static const char *handler = HANDLER;
+
+static int _subset(struct via *via)
+{
+	return VIA_T_RAID01_MIRROR(via);
+}
+
+/* Make up VIA RAID device name suffix from the serial_checksum array. */
+static uint32_t sum_serial(struct via *via)
+{
+	unsigned int i = VIA_MAX_DISKS;
+	uint32_t ret = via->array.disk_array_ex; /* FIXME: correct ? */
+
+	while (i--)
+		ret += via->serial_checksum[i];
+
+	return ret;
+}
+
+static char *_name_suffix(struct via *via)
+{
+	size_t len;
+	uint32_t sum = sum_serial(via);
+	char *ret;
+
+	if ((ret = dbg_malloc((len = snprintf(NULL, 0, "%u", sum) + 1))))
+		snprintf(ret, len, "%u", sum);
+
+	return ret;
+}
+
+/* Make up RAID device name. */
+static size_t _name(struct lib_context *lc, struct via *via, char *str,
+		     size_t len, char *suffix, unsigned int subset)
+{
+	return snprintf(str, len,
+			subset ? "via_%s-%u" : "via_%s", suffix, _subset(via));
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret, *suffix;
+	struct via *via = META(rd, via);
+
+	if (!(suffix = _name_suffix(via)))
+		return NULL;
+
+	subset = subset && (VIA_RAID_TYPE(via) == VIA_T_RAID01);
+	if ((ret = dbg_malloc((len = _name(lc, via, NULL, 0,
+					   suffix, subset) + 1)))) {
+		_name(lc, via, ret, len, suffix, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN -
+			 (subset ? 3 : 1));
+	} else
+		log_alloc_err(lc, handler);
+
+	dbg_free(suffix);
+
+	return ret;
+}
+
+/*
+ * Retrieve status of device.
+ * FIXME: is this sufficient to cover all state ?
+ */
+static enum status status(struct via *via)
+{
+	if (via->array.disk.tolerance)
+		return s_broken;
+
+	return via->array.disk.in_disk_array ? s_ok : s_undef;
+}
+
+/* Mapping of via types to generic types */
+static struct types types[] = {
+	{ VIA_T_SPAN,  t_linear },
+	{ VIA_T_RAID0, t_raid0 },
+	{ VIA_T_RAID1, t_raid1 },
+	{ VIA_T_RAID01, t_raid0 },
+        { 0, t_undef}
+};
+
+/* Neutralize disk type using generic metadata type mapping function */
+static enum type type(struct via *via)
+{
+	return rd_type(types, (unsigned int) VIA_RAID_TYPE(via));
+}
+
+/*
+ * Read a VIA RAID device
+ */
+/* Endianess conversion */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct via *via = meta;
+	unsigned int i = VIA_MAX_DISKS;
+
+	CVT16(via->signature);
+	CVT16(via->array.disk.bootable);
+	CVT32(via->array.capacity_low);
+	CVT32(via->array.capacity_high);
+	CVT32(via->array.serial_checksum);
+
+	while (i--)
+		CVT32(via->serial_checksum[i]);
+
+}
+#endif
+
+/* 8 bit checksum on first 50 bytes of metadata. */
+static uint8_t checksum(struct via *via)
+{
+	uint8_t i = 50, sum = 0;
+	
+	while (i--)
+		sum += ((uint8_t*) via)[i];
+
+printf("sum=%u via->checksum=%u\n", sum, via->checksum);
+	return sum == via->checksum;
+}
+
+static int is_via(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct via *via = meta;
+
+	if (via->signature != VIA_SIGNATURE)
+		return 0;
+
+	if (!checksum(via))
+		LOG_ERR(lc, 0, "%s: invalid checksum on %s", handler, di->path);
+
+	if (via->version_number > 1)
+		log_info(lc, "%s: version %u; format handler specified for "
+			 "version 0+1 only", handler, via->version_number);
+	
+	return 1;
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *via_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct via), VIA_CONFIGOFFSET,
+			     to_cpu, is_via, NULL, setup_rd, handler);
+}
+
+/* Decide about ordering sequence of RAID device. */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	struct via *p = META(RD(pos), via);
+	struct via *n = META(RD(new), via);
+
+	switch (VIA_RAID_TYPE(p)) {
+	case VIA_T_RAID1:
+		return VIA_T_RAID1_SOURCE(n);
+
+	default: /* span, RAID0 + RAID01 */
+		return VIA_T_RAID_INDEX(n) < VIA_T_RAID_INDEX(p);
+	}
+}
+
+/* Decide about ordering sequence of RAID subset. */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return _subset(META(RD_RS(RS(new)), via)) <
+	       _subset(META(RD_RS(RS(pos)), via));
+}
+
+static void super_created(struct raid_set *ss, void *private)
+{
+	ss->type = t_raid1;
+	ss->stride = VIA_STRIDE(META(private, via));
+}
+
+/* FIXME: handle spares in mirrors and check that types are correct. */
+static int group_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_set **ss, struct raid_dev *rd)
+{
+	struct via *via = META(rd, via);
+
+	if (!init_raid_set(lc, rs, rd, VIA_STRIDE(via),
+			   VIA_RAID_TYPE(via), handler))
+		return 0;
+
+	list_add_sorted(lc, &rs->devs, &rd->devs, dev_sort);
+
+	switch (VIA_RAID_TYPE(via)) {
+	case VIA_T_SPAN:
+	case VIA_T_RAID0:
+	case VIA_T_RAID1:
+		if (!find_set(lc, NULL, rs->name, FIND_TOP))
+			list_add_tail(&rs->list, LC_RS(lc));
+
+		break;
+
+	case VIA_T_RAID01:
+		/* Sort RAID disk into appropriate subset. */
+		if (!(*ss = join_superset(lc, name, super_created,
+					  set_sort, rs, rd)))
+			return 0;
+	}
+
+	return 1;
+}
+
+/* Add a VIA RAID device to a set */
+static struct raid_set *via_group(struct lib_context *lc,
+					 struct raid_dev *rd)
+{
+	struct raid_set *rs, *ss = NULL;
+
+	if (T_SPARE(rd))
+		return NULL;
+
+	if ((rs = find_or_alloc_raid_set(lc, rd->name, FIND_ALL, rd,
+					 NO_LIST, NO_CREATE, NO_CREATE_ARG)))
+		return group_rd(lc, rs, &ss, rd) ? (ss ? ss : rs) : NULL;
+
+	return NULL;
+}
+
+static int via_write(struct lib_context *lc, struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct via *via = META(rd, via);
+
+	to_disk(via);
+#endif
+	ret = write_metadata(lc, handler, rd, -1, erase);
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(via);
+#endif
+
+	return ret;
+}
+
+/*
+ * Check a VIA RAID set.
+ *
+ * FIXME: more sanity checks.
+ */
+/* Figure total number of disks depending on RAID type. */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	struct via *via = META(rd, via);
+
+	return VIA_RAID_TYPE(via) == VIA_T_RAID1 ? 2 : VIA_RAID_DISKS(via);
+}
+
+static int check_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_dev *rd, void *context)
+{
+	struct via *via = META(rd, via);
+
+	log_dbg(lc, "checking %s device \"%s\"", handler, rd->di->path);
+	if (VIA_ARRAY_INDEX(via) != VIA_ARRAY_INDEX(via))
+		LOG_ERR(lc, 0, "%s: array index wrong on %s for set \"%s\"",
+			handler, rd->di->path, rs->name);
+
+	if (VIA_RAID_TYPE(via) != VIA_RAID_TYPE(via))
+		LOG_ERR(lc, 0, "%s: RAID type wrong on %s for set \"%s\"",
+			handler, rd->di->path, rs->name);
+
+	return 1;
+}
+
+static int via_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return check_raid_set(lc, rs, devices, NULL, check_rd, NULL, handler);
+}
+
+/*
+ * IO error event handler.
+ */
+static int event_io(struct lib_context *lc, struct event_io *e_io)
+{
+	struct raid_dev *rd = e_io->rd;
+	struct via *via = META(rd, via);
+
+	/* Avoid write trashing. */
+	if (status(via) & s_broken)
+		return 0;
+
+	via->array.disk.tolerance = 1;
+
+	return 1;
+}
+
+static struct event_handlers via_event_handlers = {
+	.io = event_io,
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about the RAID device.
+ */
+static void via_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	unsigned int i;
+	struct via *via = META(rd, via);
+
+	log_print(lc, "%s (%s):", rd->di->path, handler);
+	DP("signature: 0x%x", via, via->signature);
+	DP("version_number: %u", via, via->version_number);
+	P("array.disk.bootable: %u", via, via->array.disk,
+	  via->array.disk.bootable);
+	P("array.disk.enable_enhanced: %u", via, via->array.disk,
+	  via->array.disk.enable_enhanced);
+	P("array.disk.in_disk_array: %u", via, via->array.disk,
+	  via->array.disk.in_disk_array);
+	P("array.disk.raid_type: %u", via, via->array.disk,
+	  VIA_RAID_TYPE(via));
+	P("array.disk.array_index: %u", via, via->array.disk,
+	  VIA_ARRAY_INDEX(via));
+
+	P("array.disk.raid_type_info: %u", via, via->array.disk,
+	  via->array.disk.raid_type_info);
+	P("array.disk.raid_type_info(INDEX): %u", via, via->array.disk,
+	  VIA_T_RAID_INDEX(via));
+	P("array.disk.raid_type_info(MIRROR): %u", via, via->array.disk,
+	  VIA_T_RAID01_MIRROR(via));
+	P("array.disk.raid_type_info(DIRTY): %u", via, via->array.disk,
+	  VIA_T_RAID01_DIRTY(via));
+
+	P("array.disk.tolerance: %u", via, via->array.disk,
+	  via->array.disk.tolerance);
+	DP("array.disk_array_ex: 0x%x", via, via->array.disk_array_ex);
+	DP("array.capacity_low: %u", via, via->array.capacity_low);
+	DP("array.capacity_high: %u", via, via->array.capacity_high);
+	DP("array.serial_checksum: %u", via, via->array.serial_checksum);
+
+	for (i = 0; i < VIA_MAX_DISKS; i++)
+		P2("serial_checksum[%u]: %u", via, i, via->serial_checksum[i]);
+
+	DP("checksum: %u", via, via->checksum);
+}
+#endif
+
+static struct dmraid_format via_format = {
+	.name	= HANDLER,
+	.descr	= "VIA Software RAID",
+	.caps	= "S,0,1,10",
+	.format = FMT_RAID,
+	.read	= via_read,
+	.write	= via_write,
+	.group	= via_group,
+	.check	= via_check,
+	.events	= &via_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= via_log,
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_via(struct lib_context *lc)
+{
+	return register_format_handler(lc, &via_format);
+}
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct via *via = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = VIA_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*via);
+	rd->meta_areas->area = (void*) via;
+
+        rd->di = di;
+	rd->fmt = &via_format;
+
+	rd->status = status(via);
+	rd->type   = type(via);
+
+	rd->offset = VIA_DATAOFFSET;
+	rd->sectors = rd->meta_areas->offset;
+
+        return (rd->name = name(lc, rd, 1)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/ataraid/via.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/ataraid/via.h
+++ -	2008-02-22 16:57:42.636569000 +0000
@@ -0,0 +1,77 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file DISCLAIMER at the top of this source tree for license information.
+ */
+
+#ifndef	_VIA_H_
+#define	_VIA_H_
+
+#ifdef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+/* VIA metadata and data  offsets. */
+#define	VIA_CONFIGOFFSET	((di->sectors - 1) << 9)
+#define	VIA_DATAOFFSET		0
+
+#define	VIA_MAX_DISKS		8
+
+struct disk {
+	uint16_t	bootable:1;		/* BIOS boot */
+	uint16_t	enable_enhanced:1;	/* Unused */
+	uint16_t	in_disk_array:1;	/* Used/Spare */
+	uint16_t	raid_type:4;
+#define	VIA_T_RAID0	0
+#define	VIA_T_RAID1	1
+#define	VIA_T_SPAN	8
+#define	VIA_T_RAID01	9
+#define VIA_RAID_TYPE(x)	((x)->array.disk.raid_type)
+	uint16_t	array_index:3;
+#define VIA_ARRAY_INDEX(x)	((x)->array.disk.array_index)
+	uint16_t	raid_type_info:5;
+/* SPAN + RAID 0 */
+#define	VIA_T_RAID_INDEX(x) ((x)->array.disk.raid_type_info & 0x7)
+
+/* RAID 1 */
+#define	VIA_T_RAID1_SOURCE(x) (((x)->array.disk.raid_type_info & 0x3) == 0)
+#define	VIA_T_RAID1_SPARE(x)  (((x)->array.disk.raid_type_info & 0x3) == 1)
+#define	VIA_T_RAID1_MIRROR(x) (((x)->array.disk.raid_type_info & 0x3) == 2)
+#define	VIA_T_RAID1_DIRTY(x)  (((x)->array.disk.raid_type_info & 0x4) >> 2)
+
+/* RAID 0+1 */
+// #define	VIA_T_RAID01_INDEX(x)	VIA_T_RAID_INDEX(x)
+#define	VIA_T_RAID01_MIRROR(x)	(((x)->array.disk.raid_type_info & 0x8) >> 3)
+#define	VIA_T_RAID01_DIRTY(x)	(((x)->array.disk.raid_type_info & 0x10) >> 4)
+
+/* SPAN */
+#define	VIA_T_SPAN_INDEX(x)	((x)->array.disk.raid_type_info & 0x7)
+	uint16_t	tolerance:1;
+} __attribute__ ((packed));
+
+struct array {
+	struct disk	disk;
+	uint8_t		disk_array_ex;
+#define	VIA_RAID_DISKS(x)	((x)->array.disk_array_ex & 0x7)
+#define	VIA_BROKEN(x)		(((x)->array.disk_array_ex & 0x8) >> 4)
+#define	VIA_STRIDE(x)		(8 << (((x)->array.disk_array_ex & 0xF0) >> 4))
+	uint32_t	capacity_low;
+	uint32_t	capacity_high;
+	uint32_t	serial_checksum;
+} __attribute__ ((packed));
+
+struct via {
+	uint16_t	signature;
+#define	VIA_SIGNATURE	0xAA55
+	uint8_t		version_number;
+	struct array	array;
+	uint32_t	serial_checksum[8];
+	uint8_t		checksum;
+} __attribute__ ((packed));
+#endif
+
+/* Prototype of the register function for this metadata format handler */
+int register_via(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/partition/dos.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/partition/dos.c
+++ -	2008-02-22 16:57:42.720563000 +0000
@@ -0,0 +1,406 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * DOS partition format handler.
+ *
+ * dos_read() and dos_group influenced by libparted.
+ */
+#define	HANDLER	"dos"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "dos.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+
+static const char *handler = HANDLER;
+
+/* Make up RAID device name. */
+static size_t _name(struct lib_context *lc, struct raid_dev *rd,
+		    unsigned short partition, char *str, size_t len,
+		    unsigned char type)
+{
+	return snprintf(str, len, type ? "%s%u" : "%s",
+			get_basename(lc, rd->di->path), partition);
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		   unsigned int part, unsigned char type)
+{
+	size_t len;
+	char *ret;
+
+	if ((ret = dbg_malloc((len = _name(lc, rd, part, NULL, 0, type) + 1))))
+		_name(lc, rd, part, ret, len, type);
+	else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/*
+ * Read a DOS partiton.
+ */
+/* Endianess conversion. */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu	NULL
+#else
+static void to_cpu(void *meta)
+{
+	struct dos *dos = meta;
+	struct dos_partition *part = dos->partitions;
+
+	for (part = dos->partitions; part < dos->partitions + 4; part++) {
+		CVT32(part->start);
+		CVT32(part->length);
+	}
+
+	CVT16(dos->magic);
+}
+#endif
+
+static int is_dos(struct lib_context *lc, struct dev_info *di, void *meta)
+{
+	struct dos *dos = meta;
+	struct dos_partition *part;
+
+	if (dos->magic != DOS_MAGIC)
+		return 0;
+
+	for (part = dos->partitions; part < &dos->partitions[4]; part++) {
+		if (part->type == PARTITION_GPT)
+			return 0;
+	}
+
+	return 1;
+}
+
+static void dos_file_metadata(struct lib_context *lc, struct dev_info *di,
+			      void *meta)
+{
+	if (OPT_DUMP(lc))
+		log_print(lc, "%s: filing metadata not supported (use fdisk "
+			  "and friends)", handler);
+}
+
+/* Allocate a DOS partition sector struct and read the data. */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *dos_read(struct lib_context *lc, struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct dos), DOS_CONFIGOFFSET,
+			     to_cpu, is_dos, dos_file_metadata,
+			     setup_rd, handler);
+}
+
+/* Support functions for dos_group to read the partition table(s). */
+static int part_is_extended (struct dos_partition *part)
+{
+	return part->type == PARTITION_EXT ||
+	       part->type == PARTITION_EXT_LBA ||
+	       part->type == PARTITION_LINUX_EXT;
+}
+
+/* Get a partition start offset relative to a base location. */
+static uint64_t get_part_start(const struct dos_partition *raw_part,
+			     uint64_t offset)
+{
+	return (uint64_t) raw_part->start + offset;
+}
+
+/* RAID set allocation support function. */
+static struct raid_set *_alloc_raid_set(struct lib_context *lc,
+					struct raid_dev *rd)
+{
+	struct raid_set *rs;
+
+	if ((rs = find_set(lc, NULL, rd->name, FIND_TOP)))
+		LOG_ERR(lc, NULL, "%s: RAID set %s already exists",
+			handler, rs->name);
+
+	if (!(rs = alloc_raid_set(lc, handler)))
+		return NULL;
+
+	rs->status = rd->status;
+	rs->type   = rd->type;
+
+       	if (!(rs->name = dbg_strdup(rd->name))) {
+		dbg_free(rs);
+		rs = NULL;
+		log_alloc_err(lc, handler);
+	}
+
+	return rs;
+}
+
+/*
+ * Allocate a DOS RAID device and a set.
+ * Set the device up and add it to the set.
+ */
+static int _create_rs_and_rd(struct lib_context *lc, struct raid_dev *rd,
+			     struct dos_partition *raw_part, uint64_t sector,
+			     unsigned int part)
+{
+	struct raid_dev *r;
+	struct raid_set *rs;
+
+	if (!(r = alloc_raid_dev(lc, handler)))
+		return 0;
+
+	if (!(r->di = alloc_dev_info(lc, rd->di->path)))
+		goto free_raid_dev;
+
+        if (!(r->name = name(lc, rd, part, 1)))
+		goto free_di;
+
+	r->fmt = rd->fmt;
+	r->status = rd->status;
+	r->type = rd->type;
+
+	if ((uint64_t) raw_part->start > sector)
+		sector = 0;
+
+	r->offset = get_part_start(raw_part, sector);
+	r->sectors = (uint64_t) raw_part->length;
+
+	if (!(rs = _alloc_raid_set(lc, r)))
+		goto free_di;
+
+	list_add_tail(&r->devs, &rs->devs);
+	list_add_tail(&rs->list, LC_RS(lc));
+
+	return 1;
+
+  free_di:
+	free_dev_info(lc, r->di);
+  free_raid_dev:
+	free_raid_dev(lc, &r);
+
+	return 0;
+}
+
+/*
+ * RAID set grouping.
+ *
+ * In this case it is not really about grouping, more about providing
+ * the propper input to the activation layer by creating a RAID set per
+ * partition and a RAID device hanging off it mapping the partition linearly.
+ *
+ * Partition code inspired by libparted and squeezed for this purpose (lemon).
+ */
+/* FIXME: Check for position of partition */
+static int is_partition(struct dos_partition *p, uint64_t start_sector)
+{
+	return p->type != PARTITION_EMPTY && p->length && p->start;
+}
+
+static int group_rd_extended(struct lib_context *lc, struct raid_dev *rd,
+			     uint64_t start_sector, uint64_t *extended_root,
+			     unsigned int part)
+{
+	int ret = 0;
+	uint64_t new_start_sector;
+	struct dos *dos;
+	struct dos_partition *p1, *p2;
+
+	/* Allocate and read a logical partition table. */
+	if (!(dos = alloc_private_and_read(lc, handler, sizeof(*dos),
+					   rd->di->path, start_sector << 9)))
+		return 0;
+
+	/* Weird: empty extended partitions are filled with 0xF6 by PM. */
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	to_cpu(dos);
+#endif
+	if (dos->magic == PARTITION_MAGIC_MAGIC)
+		goto out;
+	
+	/* Check magic to see if this is a real partition table. */
+	if (dos->magic != DOS_MAGIC)
+		goto out;
+
+	/*
+	 * Logical partition tables only have two entries,
+	 * one for the partition and one for the next partition table.
+	 */
+	 
+	/*
+	 * An entry pointing to the present logical partition.
+	 * It is an offset from the present partition table location.
+	 */
+	p1 = &dos->partitions[0];
+	
+	/*
+	 * An entry pointing to the next logical partition table.
+	 * It is an offset from the main extended partition start.
+	 */
+	p2 = &dos->partitions[1];
+
+	/* If it is a partition, add it to the set */
+	if (is_partition(p1, start_sector) &&
+	    !_create_rs_and_rd(lc, rd, p1, start_sector, part++))
+		goto out;
+
+	/*
+	 * Recurse into the logical partition chain.
+	 * Save start of main extended partition since
+	 * logical partition tables in the extended partition
+	 * are at offsets from this.
+	 */
+	if (!*extended_root)
+		*extended_root = start_sector;
+	new_start_sector = get_part_start(p2, *extended_root);
+	
+	if (is_partition(p2, start_sector) &&
+	    !group_rd_extended(lc, rd, new_start_sector, extended_root, part))
+		goto out;
+
+	ret = 1;
+
+  out:
+	dbg_free(dos);
+	return ret;
+}
+
+/* Handle primary partitions. */
+static int group_rd(struct lib_context *lc, struct raid_dev *rd,
+		    uint64_t start_sector)
+{
+	unsigned int i;
+	uint64_t part_start, part_end,
+		 extended_part_start = 0, extended_root = 0;
+	struct dos *dos = META(rd, dos);
+	struct dos_partition *raw_table_entry;
+
+	/*
+	 * Walk the 4 array entries in the primary partition table.
+	 *
+	 * Finish all primary partitions before going on to
+	 * the extended partition. Hopefully, for now you only
+	 * have one extended partition!
+	 */
+	for (i = 0; i < 4; i++) {
+		raw_table_entry = &dos->partitions[i];
+		if (!is_partition(raw_table_entry, start_sector))
+			continue;
+
+		/* Add partition start from partition table to
+		 * start of drive.
+		 */
+		part_start = get_part_start(raw_table_entry, start_sector);
+		part_end   = part_start + raw_table_entry->length;
+		
+		/* Avoid infinite recursion (mostly). */
+		if (part_start == start_sector ||
+		    part_end > rd->sectors)
+			continue;
+
+		/*
+		 * If we have an extended partition, save
+		 * partition start as a flag for later use.
+		 * Else go put partition in set.
+		 */
+		if (part_is_extended(raw_table_entry))
+			extended_part_start = part_start;
+		else if (!_create_rs_and_rd(lc, rd, raw_table_entry,
+					    start_sector, i + 1))
+			return 0;
+	}
+	
+	/* When we are finished with all the primary partitions,
+	 * go do the extended partition if we have one.
+	 * It always starts with partition 5.
+	 */
+	return extended_part_start ?
+	       group_rd_extended(lc, rd, extended_part_start,
+				 &extended_root, 5) : 1;
+}
+
+/* Add a DOS RAID device to a set */
+static struct raid_set *dos_group(struct lib_context *lc,
+				    struct raid_dev *rd)
+{
+	/*
+	 * Once we get here, a DOS partition table
+	 * has been discovered by dos_read.
+	 *
+	 * We need to run through the (nested) partition tables and create
+	 * a RAID set and a linear RAID device hanging off it for every primary
+	 * and logical partition, so that the activate code is happy.
+	 *
+	 * We start with start_sector = 0 because all primary partitions are
+	 * located at offsets from the start of the drive. This COULD be changed
+	 * to something else for some strange partitioning scheme because the
+	 * code will handle it.
+	 */
+	return group_rd(lc, rd, 0) ? (struct raid_set*) 1 : NULL;
+}
+
+/*
+ * Check integrity of a DOS RAID set.
+ */
+static int dos_check(struct lib_context *lc, struct raid_set *rs)
+{
+	return 1; /* Nice, eh ? */
+}
+
+static struct dmraid_format dos_format = {
+	.name	= HANDLER,
+	.descr	= "DOS partitions on SW RAIDs",
+	.caps	= NULL, /* Not supported */
+	.format = FMT_PARTITION,
+	.read	= dos_read,
+	.write	= NULL, /* Not supported */
+	.group	= dos_group,
+	.check	= dos_check,
+	.events	= NULL, /* Not supported */
+#ifdef DMRAID_NATIVE_LOG
+	.log	= NULL, /* Not supported */
+#endif
+};
+
+/* Register this format handler with the format core. */
+int register_dos(struct lib_context *lc)
+{
+	return register_format_handler(lc, &dos_format);
+}
+
+/*
+ * Set the RAID device contents up derived from the DOS ones
+ *
+ * For a DOS partition we essentially just save the
+ * partition table sector and let dos_group do the rest...
+ */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct dos *dos = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = DOS_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*dos);
+	rd->meta_areas->area = (void*) dos;
+
+        rd->di  = di;
+	rd->fmt = &dos_format;
+
+        rd->status = s_ok; /* Always :-) */
+	rd->type   = t_partition;
+
+	rd->offset   = DOS_DATAOFFSET;
+	rd->sectors  = di->sectors;
+
+        return (rd->name = name(lc, rd, 0, 0)) ? 1 : 0;
+}
/cvs/dm/dmraid/lib/format/partition/dos.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/partition/dos.h
+++ -	2008-02-22 16:57:42.804765000 +0000
@@ -0,0 +1,58 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * DOS partition defintion.
+ *
+ * Profited from libparted.
+ */
+
+#ifndef	_DOS_H_
+#define	_DOS_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+#include <stdint.h>
+
+#define	DOS_CONFIGOFFSET	0
+#define	DOS_DATAOFFSET		0
+
+struct chs {
+        uint8_t         head;
+        uint8_t         sector;
+        uint8_t         cylinder;
+} __attribute__ ((packed));
+
+struct dos_partition {
+        uint8_t		boot_ind;	/* 00:  0x80 - active */
+	struct chs	chs_start;	/* 01: */
+	uint8_t		type;		/* 04: partition type */
+#define PARTITION_EMPTY		0x00
+#define PARTITION_EXT		0x05
+#define PARTITION_EXT_LBA	0x0f
+#define PARTITION_LINUX_EXT	0x85
+#define PARTITION_GPT		0xee
+	struct chs	chs_end;	/* 05: */
+	uint32_t	start;		/* 08: starting sector from 0 */
+	uint32_t	length;		/* 0c: nr of sectors in partition */
+} __attribute__ ((packed));
+
+struct dos {
+	uint8_t			boot_code [446];
+	struct dos_partition	partitions [4];
+	uint16_t		magic;
+#define	DOS_MAGIC	0xAA55
+#define	PARTITION_MAGIC_MAGIC	0xF6F6
+} __attribute__ ((packed));
+
+#endif /* FORMAT_HANDLER */
+
+/* Prototype of the register function for this metadata format handler */
+int register_dos(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/format/template/README,v  -->  standard output
revision 1.1
--- dmraid/lib/format/template/README
+++ -	2008-02-22 16:57:42.892880000 +0000
@@ -0,0 +1,13 @@
+This directory contains a dmraid library format handler template.
+
+See template.h for a basic header defining the ondisk metadata structures
+and template.c showing the basic code structure of a format handler.
+Watch the CODEME remarks for coding areas to cover.
+You probably want to look at lib/format/ataraid/*.[ch] as well ;)
+
+Put your format handler .h and .c files into an appropriate (new)
+subdirectory (eg, another ATARAID format handler goes into lib/format/ataraid),
+add your register function to lib/format/register.h, add your header file
+to lib/format/ondisk.h and add the .c file to SOURCES in lib/Makefile.in
+(don't forget to run configure before make).
+
/cvs/dm/dmraid/lib/format/template/template.c,v  -->  standard output
revision 1.1
--- dmraid/lib/format/template/template.c
+++ -	2008-02-22 16:57:42.978877000 +0000
@@ -0,0 +1,230 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Template to implement metadata format handlers.
+ */
+
+#define	HANDLER	"template"
+
+#include "internal.h"
+#define	FORMAT_HANDLER
+#include "template.h"
+
+#if	BYTE_ORDER != LITTLE_ENDIAN
+#  define	DM_BYTEORDER_SWAB
+#  include	<datastruct/byteorder.h>
+#endif
+
+static const char *handler = HANDLER;
+
+/* Make up RAID device name. */
+/* CODEME: implement creation of senseful name for the RAID device */
+static size_t _name(struct template *template, char *str, size_t len,
+		    unsigned int subset)
+{
+	return snprintf(str, len, "template");
+}
+
+static char *name(struct lib_context *lc, struct raid_dev *rd,
+		  unsigned int subset)
+{
+	size_t len;
+	char *ret;
+	struct template *template = META(rd, template);
+
+	if ((ret = dbg_malloc((len = _name(template, NULL, 0, subset) + 1)))) {
+		_name(template, ret, len, subset);
+		mk_alpha(lc, ret + HANDLER_LEN, len - HANDLER_LEN);
+	} else
+		log_alloc_err(lc, handler);
+
+	return ret;
+}
+
+/* Mapping of template types to generic types */
+/*
+ * CODEME: mappings of template private level types to generic ones 
+ * (see metadata.h for generic ones)
+ */
+static struct types types[] = {
+	{ TEMPLATE_T_SPAN,  t_linear },
+	{ TEMPLATE_T_RAID0, t_raid0 },
+        { 0, t_undef}
+};
+
+/* Neutralize disk type using generic metadata type mapping function */
+static enum type template_type(struct lib_context *lc, struct raid_dev *rd)
+{
+	return rd_type(types, (unsigned int) (META(rd, template))->type);
+}
+
+/*
+ * Read a Template RAID device
+ */
+/* CODEME: endianess conversion */
+#if	BYTE_ORDER == LITTLE_ENDIAN
+#  define	to_cpu(x)
+#else
+static void to_cpu(struct template *template)
+{
+	CVT32(template->something);
+	...
+}
+#endif
+
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info);
+static struct raid_dev *template_read(struct lib_context *lc,
+					struct dev_info *di)
+{
+	return read_raid_dev(lc, di, NULL,
+			     sizeof(struct template), TEMPLATE_CONFIGOFFSET,
+			     NULL, NULL, NULL, handler);
+}
+
+/*
+ * Decide about ordering sequence of RAID device.
+ * (Called by list_add_sorted().
+ */
+static int dev_sort(struct list_head *pos, struct list_head *new)
+{
+	return (META(RD(new), template))->disk_number <
+	       (META(RD(pos), template))->disk_number;
+}
+
+/*
+ * Decide about ordering sequence of RAID device.
+ * (Called by join_superset().
+ */
+static int set_sort(struct list_head *pos, struct list_head *new)
+{
+	return _subset(META(RD_RS(RS(new)), via)) <
+	       _subset(META(RD_RS(RS(pos)), via));
+}
+
+/* Add a Template RAID device to a set */
+static struct raid_set *template_group(struct lib_context *lc,
+					 struct raid_dev *rd)
+{
+	if (T_SPARE(rd))
+		return NULL;
+
+	/* CODEME: add grouping logic
+	 *
+	 * This involves list_add_sorted() and in case of
+	 * stacked RAID sets, join_superset() calls as well.
+	 */
+	log_err(lc, "%s: implement grouping logic for RAID set", handler);
+	return NULL;
+}
+
+/* CODEME: Write private RAID metadata to device */
+static int template_write(struct lib_context *lc,
+			  struct raid_dev *rd, int erase)
+{
+	int ret;
+#if	BYTE_ORDER != LITTLE_ENDIAN
+	struct template *template = META(rd, template);
+#endif
+
+	/* CODEME: in case there's more complex metadata areas */
+	to_disk(template);
+	ret = write_metadata(lc, handler, rd, -1, erase);
+	to_cpu(template);
+	return ret;
+}
+
+/*
+ * Check integrity of a RAID set.
+ */
+static unsigned int devices(struct raid_dev *rd, void *context)
+{
+	LOG_ERR(lc, 0, "%s: implement RAID device # function", handler);
+}
+
+static int check_rd(struct lib_context *lc, struct raid_set *rs,
+		    struct raid_dev *rd, void *context);
+{
+	LOG_ERR(lc, 0, "%s: implement RAID device integrity checks", handler);
+}
+
+static int template_check(struct lib_context *lc, struct raid_set *rs)
+{
+	/* CODEME: implement */
+	return check_raid_set(lc, rs, devices, devices_context,
+			      check_rd, check_rd_context, handler);
+}
+
+static struct event_handlers template_event_handlers = {
+	.io = event_io,	/* CODEME: */
+	.rd = NULL,	/* FIXME: no device add/remove event handler yet. */
+};
+
+#ifdef DMRAID_NATIVE_LOG
+/*
+ * Log native information about the RAID device.
+ */
+static void template_log(struct lib_context *lc, struct raid_dev *rd)
+{
+	struct template *template = META(rd, template);
+
+	/* CODEME: implement (use P(), ... macors from format.h */
+	log_print(lc, "%s: implement displaying metadata variables", handler);
+}
+#endif
+
+static struct dmraid_format template_format = {
+	.name	= HANDLER,
+	.descr	= "Template RAID",
+	.caps	= "(Insert RAID levels here)",
+	.format = FMT_RAID,
+	.read	= template_read,
+	.write	= template_write,
+	.group	= template_group,
+	.check	= template_check,
+	.events	= &template_event_handlers,
+#ifdef DMRAID_NATIVE_LOG
+	.log	= template_log,
+#endif
+};
+
+/* Register this format handler with the format core */
+int register_template(struct lib_context *lc)
+{
+	return register_format_handler(lc, &template_format);
+}
+
+/* CODEME: Set the RAID device contents up derived from the TEMPLATE ones */
+static int setup_rd(struct lib_context *lc, struct raid_dev *rd,
+		    struct dev_info *di, void *meta, union read_info *info)
+{
+	struct template *template = meta;
+
+	if (!(rd->meta_areas = alloc_meta_areas(lc, rd, handler, 1)))
+		return 0;
+
+	rd->meta_areas->offset = TEMPLATE_CONFIGOFFSET >> 9;
+	rd->meta_areas->size = sizeof(*template);
+	rd->meta_areas->area = (void*) template;
+
+        rd->di = di;
+	rd->fmt = &template_format;
+
+	rd->status = s_ok; /* CODEME: derive from metadata. */
+	rd->type   = template_type(template);
+
+	rd->offset = TEMPLATE_DATAOFFSET;
+	/* CODEME: correct sectors. */
+	rd->sectors = rd->meta_areas->offset;
+
+
+        if ((rd->name = name(lc, rd, 1)))
+		return 1;
+
+	return 0;
+}
/cvs/dm/dmraid/lib/format/template/template.h,v  -->  standard output
revision 1.1
--- dmraid/lib/format/template/template.h
+++ -	2008-02-22 16:57:43.061577000 +0000
@@ -0,0 +1,42 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_TEMPLATE_H_
+#define	_TEMPLATE_H_
+
+#ifdef	FORMAT_HANDLER
+#undef	FORMAT_HANDLER
+
+/* CODEME: code ondisk metadata definitions */
+
+#include <stdint.h>
+
+#define	TEMPLATE_CONFIGOFFSET	((di->sectors - 1) << 9)
+#define	TEMPLATE_DATAOFFSET		0
+
+struct template {
+	uint8_t magic_name[8];	/* metadata has a 'magic' name ? */
+#define	TEMPLATE_MAGIC_NAME	"TEMPLATE"
+
+	uint32_t magic;		/* and/or metadata has a 'magic' number ? */
+#define	TEMPLATE_MAGIC_OK	0xABCDEF
+
+	uint8_t	type;		/* RAID level */
+#define	TEMPLATE_T_SPAN	0
+#define	TEMPLATE_T_RAID0 1
+
+	uint8_t disk_number;	/* Absolute disk number in set. */
+	/* etc. */
+
+	.......			/* members for numbers of disks, whatever... */
+} __attribute__ ((packed));
+#endif
+
+/* Prototype of the register function for this metadata format handler */
+int register_template(struct lib_context *lc);
+
+#endif
/cvs/dm/dmraid/lib/locking/locking.c,v  -->  standard output
revision 1.1
--- dmraid/lib/locking/locking.c
+++ -	2008-02-22 16:57:43.158050000 +0000
@@ -0,0 +1,115 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <errno.h>
+
+#ifndef __KLIBC__
+# include <sys/file.h>
+#endif
+
+#include "internal.h"
+
+/* File locking private data. */
+static const char *lock_file = "/var/lock/dmraid/.lock";
+static int lf = -1;
+
+/* flock file. */
+static int lock(struct lib_context *lc, struct resource *res)
+{
+	/* Already locked. */
+	if (lf > -1)
+		return 1;
+
+	log_warn(lc, "locking %s", lock_file);
+	if ((lf = open(lock_file, O_CREAT | O_APPEND | O_RDWR, 0777)) < 0)
+		LOG_ERR(lc, 0, "opening lockfile %s", lock_file);
+
+	if (flock(lf, LOCK_EX)) {
+		close(lf);
+		lf = -1;
+		LOG_ERR(lc, 0, "flock lockfile %s", lock_file);
+	}
+
+	return 1;
+}
+
+/* Unlock file. */
+static void unlock(struct lib_context *lc, struct resource *res)
+{
+	/* Not locked! */
+	if (lf == -1)
+		return;
+
+	log_warn(lc, "unlocking %s", lock_file);
+	unlink(lock_file);
+	if (flock(lf, LOCK_NB | LOCK_UN))
+		log_err(lc, "flock lockfile %s", lock_file);
+
+	if (close(lf))
+		log_err(lc, "close lockfile %s", lock_file);
+
+	lf = -1;
+}
+
+/* File base locking interface. */
+static struct locking file_locking = {
+	.name	 = "file",
+	.lock 	 = lock,
+	.unlock	 = unlock,
+};
+
+static int init_file_locking(struct lib_context *lc)
+{
+	int ret = 0;
+	char *dir;
+
+	if (!(dir = get_dirname(lc, (char*) lock_file)))
+		return 0;
+
+	if (!mk_dir(lc, dir))
+		goto out;
+
+	/* Fail on read-only file system. */
+	if (access(dir, R_OK | W_OK) && errno == EROFS)
+		goto out;
+
+	lc->lock = &file_locking;
+	ret = 1;
+
+   out:
+	dbg_free(dir);
+
+	return ret;
+}
+
+/*
+ * External locking interface.
+ */
+
+/* Initialize locking. */
+int init_locking(struct lib_context *lc)
+{
+	if (OPT_IGNORELOCKING(lc))
+		return 1;
+
+	if (lc->locking_name)
+		BUG(lc, 0, "no locking selection yet");
+
+	return init_file_locking(lc);
+}
+
+/* Hide locking. */
+int lock_resource(struct lib_context *lc, struct resource *res)
+{
+	return OPT_IGNORELOCKING(lc) ? 1 : lc->lock->lock(lc, res);
+}
+
+/* Hide unlocking. */
+void unlock_resource(struct lib_context *lc, struct resource *res)
+{
+	return OPT_IGNORELOCKING(lc) ? 1 : lc->lock->unlock(lc, res);
+}
/cvs/dm/dmraid/lib/log/log.c,v  -->  standard output
revision 1.1
--- dmraid/lib/log/log.c
+++ -	2008-02-22 16:57:43.247896000 +0000
@@ -0,0 +1,59 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <stdarg.h>
+#include "internal.h"
+
+static const char *_prefixes[] = {
+	NULL,
+	"INFO",
+	"NOTICE",
+	"WARN",
+	"DEBUG",
+	"ERROR",
+	"FATAL",
+};
+
+static const char *_prefix(int level)
+{
+	return level < ARRAY_SIZE(_prefixes) ? _prefixes[level] : "UNDEF";
+}
+
+void plog(struct lib_context *lc, int level, int lf, const char *file,
+	  int line, const char *format, ...)
+{
+	int o = LC_VERBOSE, l = level;
+	FILE *f = stdout;
+	va_list ap;
+
+ 	if (level == _PLOG_DEBUG) {
+		o = LC_DEBUG;
+		l -= _PLOG_WARN;
+	}
+
+	if (level == _PLOG_ERR || level == _PLOG_FATAL)
+		f = stderr;
+	/* Checking lc here to allow early calls without a context. */
+	else if (lc && lc_opt(lc, o) < l)
+		return;
+
+	if (_prefix(level))
+		fprintf(f, "%s: ", _prefix(level));
+
+	va_start(ap, format);
+	vfprintf(f, format, ap);
+	va_end(ap);
+
+	if (lf)
+		fputc('\n', f);
+}
+
+/* This is used so often in the metadata format handlers and elsewhere. */
+int log_alloc_err(struct lib_context *lc, const char *who)
+{
+	LOG_ERR(lc, 0, "%s: allocating", who);
+}
/cvs/dm/dmraid/lib/log/log.h,v  -->  standard output
revision 1.1
--- dmraid/lib/log/log.h
+++ -	2008-02-22 16:57:43.364058000 +0000
@@ -0,0 +1,91 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _LOG_H_
+#define _LOG_H_
+
+#include <stdio.h>
+
+/* Compatibility type for logging. */
+#define	PRIzu	"zu"
+
+/* Log levels. */
+#define _PLOG_INFO 1
+#define _PLOG_NOTICE 2
+#define _PLOG_WARN 3
+#define _PLOG_DEBUG 4
+#define _PLOG_ERR 5
+#define _PLOG_FATAL 6
+
+struct lib_context;
+void plog(struct lib_context *lc, int level, int lf, const char *file,
+	  int line, const char *format, ...);
+int log_alloc_err(struct lib_context *lc, const char *who);
+
+#  ifdef DMRAID_MINI
+
+#define log_info(lc, x...)
+#define log_info_nnl(lc, x...)
+#define log_notice(lc, x...)
+#define log_notice_nnl(lc, x...)
+#define log_warn(lc, x...)
+#define log_warn_nn(lc, x...)
+#define log_debug(lc, x...)
+#define log_debug_nnl(lc, x...)
+#define log_dbg(lc, x...)
+#define log_dbg_nnl(lc, x...)
+
+#  else
+
+#define _log_info(lc, lf, x...) plog(lc, _PLOG_INFO, lf, __FILE__, __LINE__, x)
+#define log_info(lc, x...) _log_info(lc, 1, x)
+#define log_info_nnl(lc, x...) _log_info(lc, 0, x)
+
+#define _log_notice(lc, lf, x...) \
+	plog(lc, _PLOG_NOTICE, lf, __FILE__, __LINE__, x)
+#define log_notice(lc, x...) _log_notice(lc, 1, x)
+#define log_notice_nnl(lc, x...) _log_notice(lc, 0, x)
+
+#define _log_warn(lc, lf, x...) plog(lc, _PLOG_WARN, lf, __FILE__, __LINE__, x)
+#define log_warn(lc, x...) _log_warn(lc, 1, x)
+#define log_warn_nnl(lc, x...) _log_warn(lc, 0, x)
+
+#define _log_debug(lc, lf, x...) \
+	plog(lc, _PLOG_DEBUG, lf, __FILE__, __LINE__, x)
+#define log_debug(lc, x...) _log_debug(lc, 1, x)
+#define log_debug_nnl(lc, x...) _log_debug(lc, 0, x)
+#define log_dbg(lc, x...) log_debug(lc, x)
+#define log_dbg_nnl(lc, x...) log_debug_nnl(lc, x)
+
+#  endif
+
+#define	log_level(lc, level, x...) plog(lc, level, 1, __FILE__, __LINE__, x)
+#define	log_level_nnl(lc, level, x...) plog(lc, level, 0, __FILE__, __LINE__, x)
+
+#define _log_error(lc, lf, x...) plog(lc, _PLOG_ERR, lf, __FILE__, __LINE__, x)
+#define log_error(lc, x...) _log_error(lc, 1, x)
+#define log_error_nnl(lc, x...) _log_error(lc, 0, x)
+#define log_err(lc, x...) log_error(lc, x)
+#define log_err_nnl(lc, x...) log_error_nnl(lc, x)
+
+#define _log_fatal(lc, lf, x...) \
+	plog(lc, _PLOG_FATAL, lf, __FILE__, __LINE__, x)
+#define log_fatal(lc, x...) _log_fatal(lc, 1, x)
+#define log_fatal_nnl(lc, x...) _log_fatal(lc, 0, x)
+
+
+#define	LOG_ERR(lc, ret, x...)	do { log_err(lc, x); return ret; } while (0)
+#define	BUG(lc, ret, x...)	do { LOG_ERR(lc, ret, x); } while (0)
+
+#define _log_print(lc, lf, x...) plog(lc, 0, lf, __FILE__, __LINE__, x)
+#define log_print(lc, x...)	_log_print(lc, 1, x)
+#define log_print_nnl(lc, x...)	_log_print(lc, 0, x)
+#define	LOG_PRINT(lc, ret, x...) do { log_print(lc, x); return ret; } while (0)
+#define	LOG_PRINT_NNL(lc, ret, x...) \
+	do { _log_print(lc, lc, 0, x); return ret; } while (0)
+
+#endif
/cvs/dm/dmraid/lib/metadata/metadata.c,v  -->  standard output
revision 1.1
--- dmraid/lib/metadata/metadata.c
+++ -	2008-02-22 16:57:43.478101000 +0000
@@ -0,0 +1,1086 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include "internal.h"
+#include "activate/devmapper.h"
+
+/*
+ * Type -> ascii definitions.
+ *
+ * dsp_ascii : the string used for display purposes (eg, "dmraid -s").
+ * dm_ascii  :         "       in device-mapper tables as the target keyword.
+ */
+static const struct {
+	const enum type type;
+	const char *dsp_ascii;
+	const char *dm_ascii;
+} ascii_type[] = {
+	/* enum        text         dm-target id */
+	{ t_undef,     NULL,        NULL },
+	{ t_group,     "GROUP",	    NULL },
+	{ t_partition, "partition", NULL },
+	{ t_spare,     "spare",     NULL },
+	{ t_linear,    "linear",    "linear" },
+	{ t_raid0,     "stripe",    "striped" },
+	{ t_raid1,     "mirror",    "mirror" },
+	{ t_raid4,     "raid4",     "raid45" },
+	{ t_raid5_ls,  "raid5_ls",  "raid45" },
+	{ t_raid5_rs,  "raid5_rs",  "raid45" },
+	{ t_raid5_la,  "raid5_la",  "raid45" },
+	{ t_raid5_ra,  "raid5_ra",  "raid45" },
+	{ t_raid6,     "raid6",     NULL },
+};
+
+static const char *stacked_ascii_type[][5] = {
+	{ "raid10", "raid30", "raid40", "raid50", "raid60" },
+	{ "raid01", "raid03", "raid04", "raid05", "raid06" },
+};
+
+/*
+ * State definitions.
+ */
+static const struct {
+	const enum status status;
+	const char *ascii;
+} ascii_status[] = {
+	{ s_undef,        NULL },
+	{ s_setup,        "setup" },
+	{ s_broken,       "broken" },
+	{ s_inconsistent, "inconsistent" },
+	{ s_nosync,       "nosync" },
+	{ s_ok,           "ok" },
+};
+
+/* Fetch the respective ASCII string off the types array. */
+static unsigned int get_type_index(enum type type)
+{
+	unsigned int ret = ARRAY_SIZE(ascii_type);
+
+	while (ret-- && !(type & ascii_type[ret].type));
+
+	return ret;
+}
+
+const char *get_type(struct lib_context *lc, enum type type)
+{
+	return ascii_type[get_type_index(type)].dsp_ascii;
+}
+
+const char *get_dm_type(struct lib_context *lc, enum type type)
+{
+	return ascii_type[get_type_index(type)].dm_ascii;
+}
+
+/* Return the RAID type of a stacked RAID set (eg, raid10). */
+static const char *get_stacked_type(void *v)
+{
+	struct raid_set *rs = v;
+	unsigned int t = (T_RAID0(rs) ? get_type_index((RS_RS(rs))->type) :
+					get_type_index(rs->type))
+			 - get_type_index(t_raid1);
+
+	return stacked_ascii_type[T_RAID0(rs) ? 1 : 0][t];
+}
+
+/* Check, if a RAID set is stacked (ie, hierachical). */
+static inline int is_stacked(struct raid_set *rs)
+{
+	return !T_GROUP(rs) && SETS(rs);
+}
+
+/* Return the ASCII type for a RAID set. */
+const char *get_set_type(struct lib_context *lc, void *v)
+{
+	struct raid_set *rs = v;
+
+	/* Check, if a RAID set is stacked. */
+	return is_stacked(rs) ? get_stacked_type(rs) : get_type(lc, rs->type);
+}
+
+/* Fetch the respective ASCII string off the state array. */
+const char *get_status(struct lib_context *lc, enum status status)
+{
+	unsigned int i = ARRAY_SIZE(ascii_status);
+
+	while (i-- && !(status & ascii_status[i].status));
+
+	return ascii_status[i].ascii;
+}
+
+/*
+ * Calculate the size of the set by recursively summing
+ * up the size of the devices in the subsets.
+ *
+ * Pay attention to RAID > 0 types.
+ */
+static uint64_t add_sectors(struct raid_set *rs, uint64_t sectors,
+			    uint64_t add)
+{
+	add = round_down(add, rs->stride);
+
+	if (T_RAID1(rs)) {
+		if (!sectors || sectors > add)
+			sectors = add;
+	} else
+		sectors += add;
+
+	return sectors;
+}
+
+/* FIXME: proper calculation of unsymetric sets ? */
+static uint64_t smallest_disk(struct raid_set *rs)
+{
+	uint64_t ret = ~0;
+	struct raid_dev *rd;
+
+	list_for_each_entry(rd, &rs->devs, devs)
+		ret = min(ret, rd->sectors);
+
+	return ret;
+}
+
+/* Count subsets of a set. */
+static unsigned int count_subsets(struct lib_context *lc, struct raid_set *rs)
+{
+	unsigned ret = 0;
+	struct raid_set *r;
+
+	list_for_each_entry(r, &rs->sets, list)
+		ret++;
+
+	return ret;
+}
+
+/* Calculate total sectors of a (hierarchical) RAID set. */
+uint64_t total_sectors(struct lib_context *lc, struct raid_set *rs)
+{
+        uint64_t sectors = 0;
+        struct raid_dev *rd;
+
+	/* Stacked RAID sets. */
+	if (!T_GROUP(rs)) {
+        	struct raid_set *r;
+
+		list_for_each_entry(r, &rs->sets, list)
+			sectors = add_sectors(rs, sectors,
+					      total_sectors(lc, r));
+	}
+
+	/* RAID device additions taking size maximization into account. */
+	if (DEVS(rs)) {
+		uint64_t min = F_MAXIMIZE(rs) ? 0 : smallest_disk(rs);
+
+		list_for_each_entry(rd, &rs->devs, devs) {
+			if (!T_SPARE(rd))
+				sectors = add_sectors(rs, sectors,
+						      F_MAXIMIZE(rs) ?
+						      rd->sectors : min);
+		}
+	}
+
+	/* Size correction for higher RAID levels */
+	if (T_RAID4(rs) || T_RAID5(rs) || T_RAID6(rs)) {
+		unsigned int i = count_subsets(lc, rs);
+		uint64_t sub = sectors / (i ? i : count_devs(lc, rs, ct_dev));
+
+		sectors -= sub;
+		if (T_RAID6(rs))
+			sectors -= sub;
+	}
+
+        return sectors;
+}
+
+/* Check if a RAID device should be counted. */
+static unsigned int _count_dev(struct raid_dev *rd, enum count_type type)
+{
+	return ((type == ct_dev && !T_SPARE(rd)) ||
+		(type == ct_spare && T_SPARE(rd)) ||
+		type == ct_all) ? 1 : 0;
+}
+
+/* Count devices in a set recursively. */
+unsigned int count_devs(struct lib_context *lc, struct raid_set *rs,
+			enum count_type count_type)
+{
+	unsigned int ret = 0;
+	struct raid_set *r;
+	struct raid_dev *rd;
+
+	list_for_each_entry(r, &rs->sets, list) {
+		if (!T_GROUP(rs))
+			ret += count_devs(lc, r, count_type);
+	}
+
+	list_for_each_entry(rd, &rs->devs, devs)
+		ret += _count_dev(rd, count_type);
+
+	return ret;
+}
+
+/*
+ * Create list of unique memory pointers of a RAID device and free them.
+ *
+ * This prevents me from having a destructor method in the metadata
+ * format handlers so far. If life becomes more complex, I might need
+ * one though...
+ */
+static void _free_dev_pointers(struct lib_context *lc, struct raid_dev *rd)
+{
+	int area, i, idx = 0;
+	void **p;
+
+	/* Count private and area pointers. */
+	if (!(area = (rd->private.ptr ? 1 : 0) + rd->areas))
+		return;
+
+	/* Allocate and initialize temporary pointer list. */
+	if (!(p = dbg_malloc(area * sizeof(*p))))
+		LOG_ERR(lc, , "allocating pointer array");
+
+	/* Add private pointer to list. */
+	if (rd->private.ptr)
+		p[idx++] = rd->private.ptr;
+
+	/* Add metadata area pointers to list. */
+	for (area = 0; area < rd->areas; area++) {
+		/* Handle multiple pointers to the same memory. */
+		for (i = 0; i < idx; i++) {
+			if (p[i] == rd->meta_areas[area].area)
+				break;
+		}
+	
+		if (i == idx)
+			p[idx++] = rd->meta_areas[area].area;
+	}
+
+	if (rd->meta_areas)
+		dbg_free(rd->meta_areas);
+
+	/* Free all RAID device pointers. */
+	while (idx--)
+		dbg_free(p[idx]);
+	
+	dbg_free(p);
+}
+
+/* Allocate dev_info struct and keep the device path */
+struct dev_info *alloc_dev_info(struct lib_context *lc, char *path)
+{
+	struct dev_info *di;
+
+	if ((di = dbg_malloc(sizeof(*di)))) {
+		if ((di->path = dbg_strdup(path)))
+			INIT_LIST_HEAD(&di->list);
+		else {
+			dbg_free(di);
+			di = NULL;
+			log_alloc_err(lc, __func__);
+		}
+	}
+
+	return di;
+}
+
+/* Free dev_info structure */
+static void _free_dev_info(struct lib_context *lc, struct dev_info *di)
+{
+	if (di->serial)
+		dbg_free(di->serial);
+
+	dbg_free(di->path);
+	dbg_free(di);
+}
+
+static inline void _free_dev_infos(struct lib_context *lc)
+{
+	struct list_head *elem, *tmp;
+
+	list_for_each_safe(elem, tmp, LC_DI(lc)) {
+		list_del(elem);
+		_free_dev_info(lc, list_entry(elem, struct dev_info, list));
+	}
+}
+
+/*
+ * Free dev_info structure or all registered
+ * dev_info structures in case di = NULL.
+ */
+void free_dev_info(struct lib_context *lc, struct dev_info *di)
+{
+	di ? _free_dev_info(lc, di) : _free_dev_infos(lc);
+}
+
+/* Allocate/Free RAID device (member of a RAID set). */
+struct raid_dev *alloc_raid_dev(struct lib_context *lc, const char *who)
+{
+	struct raid_dev *ret;
+
+	if ((ret = dbg_malloc(sizeof(*ret)))) {
+		INIT_LIST_HEAD(&ret->list);
+		INIT_LIST_HEAD(&ret->devs);
+		ret->status = s_setup;
+	} else
+		log_alloc_err(lc, who);
+
+	return ret;
+}
+
+static void _free_raid_dev(struct lib_context *lc, struct raid_dev **rd)
+{
+	struct raid_dev *r = *rd;
+
+	/* Remove if on global list. */
+	if (!list_empty(&r->list))
+		list_del(&r->list);
+
+	/*
+	 * Create list of memory pointers allocated by
+	 * the metadata format handler and free them.
+	 */
+	_free_dev_pointers(lc, r);
+
+	dbg_free(r->name);
+	dbg_free(r);
+	*rd = NULL;
+}
+
+static inline void _free_raid_devs(struct lib_context *lc)
+{
+	struct list_head *elem, *tmp;
+	struct raid_dev *rd;
+
+	list_for_each_safe(elem, tmp, LC_RD(lc)) {
+		rd = list_entry(elem, struct raid_dev, list);
+		_free_raid_dev(lc, &rd);
+	}
+}
+
+/* Free RAID device structure or all registered RAID devices if rd == NULL. */
+void free_raid_dev(struct lib_context *lc, struct raid_dev **rd)
+{
+	rd ? _free_raid_dev(lc, rd) : _free_raid_devs(lc);
+}
+
+/* Allocate/Free RAID set. */
+struct raid_set *alloc_raid_set(struct lib_context *lc, const char *who)
+{
+	struct raid_set *ret;
+
+	if ((ret = dbg_malloc(sizeof(*ret)))) {
+		INIT_LIST_HEAD(&ret->list);
+		INIT_LIST_HEAD(&ret->sets);
+		INIT_LIST_HEAD(&ret->devs);
+		ret->status = s_setup;
+		ret->type   = t_undef;
+	} else
+		log_alloc_err(lc, who);
+
+	return ret;
+}
+
+/* Free a single RAID set structure and its RAID devices. */
+static void _free_raid_set(struct lib_context *lc, struct raid_set *rs)
+{
+	struct raid_dev *rd;
+	struct list_head *elem, *tmp;
+
+	log_dbg(lc, "freeing devices of RAID set \"%s\"", rs->name);
+	list_for_each_safe(elem, tmp, &rs->devs) {
+		list_del(elem);
+		rd = RD(elem);
+
+		log_dbg(lc, "freeing device \"%s\", path \"%s\"",
+			rd->name, rd->di->path);
+
+		/* FIXME: remove partition code in favour of kpartx ? */
+		/*
+		 * Special case for partitioned sets.
+		 *
+		 * We don't hook dev_info structures for partitioned
+		 * sets up the global list, so delete them here.
+		 */
+		if (partitioned_set(lc, rs))
+			free_dev_info(lc, rd->di);
+
+		/*
+		 * We don't hook raid_dev structures for GROUP
+		 * sets up the global list, so delete them here.
+		 */
+		if (list_empty(&rd->list))
+			free_raid_dev(lc, &rd);
+	}
+
+	list_del(&rs->list);
+	dbg_free(rs->name);
+	dbg_free(rs);
+}
+
+/* Remove a set or all sets (in case rs = NULL) recursively. */
+void free_raid_set(struct lib_context *lc, struct raid_set *rs)
+{
+	struct list_head *elem, *tmp;
+
+	list_for_each_safe(elem, tmp, rs ? &rs->sets : LC_RS(lc))
+		free_raid_set(lc, RS(elem));
+
+	if (rs)
+		_free_raid_set(lc, rs);
+	else if (!list_empty(LC_RS(lc)))
+		log_fatal(lc, "lib context RAID set list not empty");
+}
+
+/* Return != 0 in case of a partitioned RAID set type. */
+int partitioned_set(struct lib_context *lc, void *rs)
+{
+	return T_PARTITION((struct raid_set*) rs);
+}
+
+/* Return != 0 in case of a partitioned base RAID set. */
+int base_partitioned_set(struct lib_context *lc, void *rs)
+{
+	return ((struct raid_set*) rs)->flags & f_partitions;
+}
+
+/* Return RAID set name. */
+const char *get_set_name(struct lib_context *lc, void *rs)
+{
+	return ((struct raid_set*) rs)->name;
+}
+
+/*
+ * Find RAID set by name.
+ *
+ * Search top level RAID set list only if where = FIND_TOP.
+ * Recursive if where = FIND_ALL.
+ */
+static struct raid_set *_find_set(struct lib_context *lc,
+				  struct list_head *list,
+				  const char *name, enum find where)
+{
+	struct raid_set *r, *ret = NULL;
+
+	log_dbg(lc, "%s: searching %s", __func__, name);
+	list_for_each_entry(r, list, list) {
+		if (!strcmp(r->name, name)) {
+			ret = r;
+			goto out;
+		}
+	}
+
+	if (where == FIND_ALL) {
+		list_for_each_entry(r, list, list) {
+			if ((ret = _find_set(lc, &r->sets, name, where)))
+				break;
+		}
+	}
+
+  out:
+	log_dbg(lc, "_find_set: %sfound %s", ret ? "" : "not ", name);
+
+	return ret;
+}
+
+struct raid_set *find_set(struct lib_context *lc,
+			  struct list_head *list,
+			  const char *name, enum find where)
+{
+	return _find_set(lc, list ? list : LC_RS(lc), name, where);
+}
+
+struct raid_set *find_or_alloc_raid_set(struct lib_context *lc,
+				char *name, enum find where,
+				struct raid_dev *rd,
+				struct list_head *list,
+				void (*f_create) (struct raid_set *super,
+						  void *private),
+				void *private)
+{
+	struct raid_set *rs;
+
+	if ((rs = find_set(lc, NULL, name, where)))
+		goto out;
+
+	if (!(rs = alloc_raid_set(lc, __func__)))
+		goto out;
+
+	if (!(rs->name = dbg_strdup(name)))
+		goto err;
+
+	if (rd && ((rs->type = rd->type), T_SPARE(rd)))
+		rs->type = t_undef;
+
+	/* If caller hands a list in, add to it. */
+	if (list)
+		list_add_tail(&rs->list, list);
+
+	/* Call any create callback. */
+	if (f_create)
+		f_create(rs, private);
+
+   out:
+	return rs;
+
+   err:
+	dbg_free(rs);
+	log_alloc_err(lc, __func__);
+
+	return NULL;
+}
+
+/* Return # of raid sets build */
+unsigned int count_sets(struct lib_context *lc, struct list_head *list)
+{
+	int ret = 0;
+	struct list_head *elem;
+
+	list_for_each(elem, list)
+		ret++;
+
+	return ret;
+}
+
+/*
+ * Count devices found
+ */
+static unsigned int _count_devices(struct lib_context *lc, enum dev_type type)
+{
+	unsigned int ret = 0;
+	struct list_head *elem, *list;
+
+	if (DEVICE & type)
+		list = LC_DI(lc);
+	else if (((RAID|NATIVE) & type))
+		list = LC_RD(lc);
+	else
+		return 0;
+	
+	list_for_each(elem, list)
+			ret++;
+
+	return ret;
+}
+
+unsigned int count_devices(struct lib_context *lc, enum dev_type type)
+{
+	return type == SET ? count_sets(lc, LC_RS(lc)) :
+			     _count_devices(lc, type);
+}
+
+/*
+ * Read RAID metadata off a device by trying
+ * all/selected registered format handlers in turn.
+ */
+static int _want_format(struct dmraid_format *fmt, const char *format,
+			enum fmt_type type)
+{
+	return fmt->format != type ||
+	       (format && strncmp(format, fmt->name, strlen(format))) ? 0 : 1;
+}
+
+static struct raid_dev *_dmraid_read(struct lib_context *lc,
+				       struct dev_info *di,
+				       struct dmraid_format *fmt)
+{
+	struct raid_dev *rd;
+
+	log_notice(lc, "%s: %-7s discovering", di->path, fmt->name);
+	if ((rd = fmt->read(lc, di))) {
+		log_notice(lc, "%s: %s metadata discovered",
+			   di->path, fmt->name);
+		rd->fmt = fmt;
+	}
+
+	return rd;
+}
+
+static struct raid_dev *dmraid_read(struct lib_context *lc,
+				      struct dev_info *di, char const *format,
+				      enum fmt_type type)
+{
+	struct format_list *fl;
+	struct raid_dev *rd = NULL, *rd_tmp;
+
+	/* FIXME: dropping multiple formats ? */
+	list_for_each_entry(fl, LC_FMT(lc), list) {
+		if (_want_format(fl->fmt, format, type) &&
+		    (rd_tmp = _dmraid_read(lc, di, fl->fmt))) {
+			if (rd) {
+				log_print(lc, "%s: \"%s\" and \"%s\" formats "
+					  "discovered (using %s)!",
+					  di->path, rd_tmp->fmt->name,
+					  rd->fmt->name, rd->fmt->name);
+				free_raid_dev(lc, &rd_tmp);
+			} else
+				rd = rd_tmp;
+		}
+	}
+
+	return rd;
+}
+
+/*
+ * Write RAID metadata to a device.
+ */
+static int dmraid_write(struct lib_context *lc,
+			struct raid_dev *rd, int erase)
+{
+	int ret = 0;
+	struct dmraid_format *fmt = rd->fmt;
+
+	if (fmt->write) {
+		log_notice(lc, "%sing metadata %s %s",
+			   erase ? "Eras" : "Writ",
+			   erase ? "on" : "to",
+			   rd->di->path);
+		ret = fmt->write(lc, rd, erase);
+	} else
+		log_err(lc, "format \"%s\" doesn't support writing metadata",
+			fmt->name);
+
+	return ret;
+}
+
+/*
+ * Group RAID device into a RAID set.
+ */
+static inline struct raid_set *dmraid_group(struct lib_context *lc,
+					    struct raid_dev *rd)
+{
+	return rd->fmt->group(lc, rd);
+}
+
+/* Check that device names are members of the devices list. */
+static int _want_device(struct dev_info *di, char **devices)
+{
+	char **dev;
+
+	if (!devices || !*devices)
+		return 1;
+
+	for (dev = devices; *dev; dev++) {
+		if (!strcmp(*dev, di->path))
+			return 1;
+	}
+
+	return 0;
+}
+
+/* Discover RAID devices. */
+void discover_raid_devices(struct lib_context *lc, char **devices)
+{
+	struct dev_info *di;
+	char *names = NULL;
+	const char delim = *OPT_STR_SEPARATOR(lc);
+
+	/* In case we've got format identifiers -> duplicate string for loop. */
+	if (OPT_FORMAT(lc) &&
+	    (!(names = dbg_strdup((char*) OPT_STR_FORMAT(lc))))) {
+		log_alloc_err(lc, __func__);
+		return;
+	}
+
+	/* Walk the list of discovered block devices. */
+	list_for_each_entry(di, LC_DI(lc), list) {
+		if (_want_device(di, devices)) {
+			char *p, *sep = names;
+			struct raid_dev *rd;
+
+			do {
+				p = sep;
+				sep = remove_delimiter(sep, delim);
+
+				if ((rd = dmraid_read(lc, di, p, FMT_RAID)))
+					list_add_tail(&rd->list, LC_RD(lc));
+
+				add_delimiter(&sep, delim);
+			} while (sep);
+		}
+	}
+
+	if (names)
+		dbg_free(names);
+}
+
+/*
+ * Discover partitions on RAID sets.
+ *
+ * FIXME: remove partition code in favour of kpartx ?
+ */
+static void _discover_partitions(struct lib_context *lc,
+				 struct list_head *rs_list)
+{
+	char *path;
+	struct dev_info *di;
+	struct raid_dev *rd;
+	struct raid_set *rs, *r;
+
+	list_for_each_entry(rs, rs_list, list) {
+		/*
+		 * t_group type RAID sets are never active!
+		 * (They are containers for subsets to activate)
+		 *
+		 * Recurse into them.
+		 */
+		if (T_GROUP(rs)) {
+			_discover_partitions(lc, &rs->sets);
+			return;
+		}
+
+		/*
+		 * Skip all "container" sets, which are not active.
+		 */
+		if (base_partitioned_set(lc, rs) ||
+		    partitioned_set(lc, rs) ||
+		    !dm_status(lc, rs))
+			continue;
+
+		log_notice(lc, "discovering partitions on \"%s\"", rs->name);
+		if (!(path = mkdm_path(lc, rs->name)))
+			return;
+
+		/* Allocate a temporary disk info struct for dmraid_read(). */
+		di = alloc_dev_info(lc, path);
+		dbg_free(path);
+		if (!di)
+			return;
+
+		di->sectors = total_sectors(lc, rs);
+		if (!(rd = dmraid_read(lc, di, NULL, FMT_PARTITION))) {
+			free_dev_info(lc, di);
+			continue;
+		}
+
+		/*
+		 * WARNING: partition group function returns
+		 * a dummy pointer because of the creation of multiple
+		 * RAID sets (one per partition) it does.
+		 *
+		 * We don't want to access that 'pointer'!
+		 */
+		if ((r = dmraid_group(lc, rd))) {
+			log_notice(lc, "created partitioned RAID set(s) for %s",
+				   di->path);
+			rs->flags |= f_partitions;
+		} else
+			log_err(lc, "adding %s to RAID set", di->path);
+
+		/*
+		 * Free the RD. We don't need it any more, because we
+		 * don't support writing partition tables.
+		 */
+		free_dev_info(lc, di);
+		free_raid_dev(lc, &rd);
+	}
+}
+
+void discover_partitions(struct lib_context *lc)
+{
+	_discover_partitions(lc, LC_RS(lc));
+}
+
+/*
+ * Group RAID set(s)
+ *
+ *	name = NULL  : build all sets
+ *	name = String: build just the one set
+ */
+static void want_set(struct lib_context *lc, struct raid_set *rs, char *name)
+{
+	if (name) {
+		size_t len1 = strlen(rs->name), len2 = strlen(name);
+
+		if (len2 > len1 ||
+		    strncmp(rs->name, name, min(len1, len2))) {
+			log_notice(lc, "dropping unwanted RAID set \"%s\"",
+				   rs->name);
+			free_raid_set(lc, rs);
+		}
+	}
+}
+
+/* Get format handler of RAID set. */
+struct dmraid_format *get_format(struct raid_set *rs)
+{
+	/* Decend RAID set hierarchy. */
+	while (SETS(rs))
+		rs = RS_RS(rs);
+
+	return DEVS(rs) ? (RD_RS(rs))->fmt : NULL;
+}
+
+/* Check metadata consistency of raid sets. */
+static void check_raid_sets(struct lib_context *lc)
+{
+	struct list_head *elem, *tmp;
+	struct raid_set *rs;
+	struct dmraid_format *fmt;
+
+	list_for_each_safe(elem, tmp, LC_RS(lc)) {
+		if (!(fmt = get_format((rs = RS(elem)))))
+			continue;
+
+		if (!fmt->check(lc, rs)) {
+			/*
+			 * FIXME: check needed if degraded activation
+			 *        is sensible.
+			 */
+			if (T_RAID1(rs))
+				log_err(lc, "keeping degraded mirror "
+					"set \"%s\"", rs->name);
+			else {
+				log_err(lc, "removing inconsistent RAID "
+					"set \"%s\"", rs->name);
+				free_raid_set(lc, rs);
+			}
+		}
+	}
+
+	return;
+}
+
+int group_set(struct lib_context *lc, char *name)
+{
+	struct raid_dev *rd;
+	struct raid_set *rs;
+
+	if (name && find_set(lc, NULL, name, FIND_TOP))
+		LOG_ERR(lc, 0, "RAID set %s already exists", name);
+
+	list_for_each_entry(rd, LC_RD(lc), list) {
+		/* FIXME: optimize dropping of unwanted RAID sets. */
+		if ((rs = dmraid_group(lc, rd))) {
+			log_notice(lc, "added %s to RAID set \"%s\"",
+				   rd->di->path, rs->name);
+			want_set(lc, rs, name);
+			continue;
+		}
+
+		if (!T_SPARE(rd))
+			log_err(lc, "adding %s to RAID set \"%s\"",
+				rd->di->path, rd->name);
+
+		/* Need to find the set and remove it. */
+		if ((rs = find_set(lc, NULL, rd->name, FIND_ALL))) {
+			log_err(lc, "removing RAID set \"%s\"", rs->name);
+			free_raid_set(lc, rs);
+		}
+	}
+
+	/* Check sanity of grouped RAID sets. */
+	check_raid_sets(lc);
+
+	return 1;
+}
+
+/* Process function on RAID set(s) */
+static void process_set(struct lib_context *lc, void *rs,
+			int (*func)(struct lib_context *lc, void *rs, int arg),
+			int arg)
+{
+	if (!partitioned_set(lc, rs))
+		func(lc, rs, arg);
+}
+
+/* FIXME: remove partition code in favour of kpartx ? */
+static void
+process_partitioned_set(struct lib_context *lc, void *rs,
+			int (*func)(struct lib_context *lc, void *rs, int arg),
+			int arg)
+{
+	if (partitioned_set(lc, rs) && !base_partitioned_set(lc, rs))
+		func(lc, rs, arg);
+}
+
+void process_sets(struct lib_context *lc,
+		  int (*func)(struct lib_context *lc, void *rs, int arg),
+		  int arg, enum set_type type)
+{
+	struct raid_set *rs;
+	void (*p)(struct lib_context *l, void *r,
+		  int (*f)(struct lib_context *lc, void *rs, int arg), int a) =
+		(type == PARTITIONS) ? process_partitioned_set : process_set;
+
+	list_for_each_entry(rs, LC_RS(lc), list)
+		p(lc, rs, func, arg);
+}
+
+/* Write RAID set metadata to devices. */
+int write_set(struct lib_context *lc, void *v)
+{
+	int ret = 1;
+	struct raid_set *r, *rs = v;
+	struct raid_dev *rd;
+
+	/* Decend hierarchy */
+	list_for_each_entry(r, &rs->sets, list) {
+		/*
+		 * FIXME: does it make sense to try the rest of the subset
+		 *	  in case we fail writing one ?
+		 */
+		if (!write_set(lc, (void*) r))
+			log_err(lc, "writing RAID subset \"%s\", continuing",
+				r->name);
+	}
+
+	/* Write metadata to the RAID devices of a set. */
+	list_for_each_entry(rd, &rs->devs, devs) {
+		/*
+		 * FIXME: does it make sense to try the rest of the
+		 *	  devices in case we fail writing one ?
+		 */
+		if (!dmraid_write(lc, rd, 0)) {
+			log_err(lc, "writing RAID device \"%s\", continuing",
+				rd->di->path);
+			ret = 0;
+		}
+	}
+
+	return ret;
+}
+
+/* Erase ondisk metadata. */
+int erase_metadata(struct lib_context *lc)
+{
+	int ret = 1;
+	struct raid_dev *rd;
+
+	list_for_each_entry(rd, LC_RD(lc), list) {
+		if (yes_no_prompt(lc, "Do you really want to erase \"%s\" "
+				  "ondisk metadata on %s",
+				  rd->fmt->name, rd->di->path) &&
+		    !dmraid_write(lc, rd, 1)) {
+			log_err(lc, "erasing ondisk metadata on %s",
+				rd->di->path);
+			ret = 0;
+		}
+	}
+
+	return ret;
+}
+
+/*
+ * Support function for metadata format handlers:
+ *
+ * Return neutralized RAID type for given mapping array (linear, raid0, ...)
+ */
+enum type rd_type(struct types *t, unsigned int type)
+{
+	for (; t->type != type && t->unified_type != t_undef; t++);
+
+	return t->unified_type;
+}
+
+/*
+ * Support function for metadata format handlers.
+ *
+ * Sort a an element into a list by optionally
+ * using a metadata format handler helper function.
+ */
+void list_add_sorted(struct lib_context *lc,
+		     struct list_head *to, struct list_head *new,
+		     int (*f_sort)(struct list_head *pos,
+				   struct list_head *new))
+{
+	struct list_head *pos;
+
+	list_for_each(pos, to) {
+		/*
+		 * Add in at the beginning of the list
+		 * (ie., after HEAD or the first entry we found),
+		 * or where the metadata format handler sort
+		 * function tells us to.
+		 */
+		if (!f_sort || f_sort(pos, new))
+			break;
+	}
+
+	/*
+	 * If we get here we either had an empty list or the sort
+	 * function hit or not -> add where pos tells us to.
+	 */
+	list_add_tail(new, pos);
+}
+
+/*
+ * Support function for format handlers:
+ *
+ * File RAID metadata and offset on device for analysis.
+ */
+/* FIXME: all files into one directory ? */
+static size_t __name(struct lib_context *lc, char *str, size_t len, char *path,
+		     const char *suffix, const char *handler)
+{
+	return snprintf(str, len, "%s_%s.%s",
+			get_basename(lc, path), handler, suffix) + 1;
+}
+
+static char *_name(struct lib_context *lc, char *path,
+		  const char *suffix, const char *handler)
+{
+	size_t len;
+	char *ret;
+
+	if ((ret = dbg_malloc((len = __name(lc, NULL, 0, path,
+					    suffix, handler)))))
+		__name(lc, ret, len, path, suffix, handler);
+	else
+		log_alloc_err(lc, __func__);
+
+	return ret;
+}
+
+static int file_data(struct lib_context *lc, const char *handler,
+		     char *path, void *data, size_t size)
+{
+	int ret = 0;
+	char *name;
+
+	if ((name = _name(lc, path, "dat", handler))) {
+		log_notice(lc, "writing metadata file \"%s\"", name);
+		ret = write_file(lc, handler, name, data, size, 0);
+		dbg_free(name);
+	}
+
+	return ret;
+}
+
+static void file_number(struct lib_context *lc, const char *handler,
+			char *path, uint64_t number, const char *suffix)
+{
+	char *name, s_number[32];
+	
+	if ((name = _name(lc, path, suffix, handler))) {
+		log_notice(lc, "writing %s to file \"%s\"", suffix, name);
+		write_file(lc, handler, name, (void*) s_number,
+		           snprintf(s_number, sizeof(s_number),
+			   "%" PRIu64 "\n", number),
+				 0);
+		dbg_free(name);
+	}
+}
+
+/*
+ * File vendor RAID metadata.
+ */
+void file_metadata(struct lib_context *lc, const char *handler,
+		   char *path, void *data, size_t size, uint64_t offset)
+{
+	if (OPT_DUMP(lc) &&
+	    file_data(lc, handler, path, data, size))
+		file_number(lc, handler, path, offset, "offset");
+}
+
+/*
+ * File RAID device size.
+ */
+void file_dev_size(struct lib_context *lc, const char *handler,
+		   struct dev_info *di)
+{
+	if (OPT_DUMP(lc))
+		file_number(lc, handler, di->path, di->sectors, "size");
+}
/cvs/dm/dmraid/lib/misc/file.c,v  -->  standard output
revision 1.1
--- dmraid/lib/misc/file.c
+++ -	2008-02-22 16:57:43.597523000 +0000
@@ -0,0 +1,101 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <errno.h>
+#include "internal.h"
+
+/* Create directory recusively. */
+static int mk_dir_recursive(struct lib_context *lc, const char *dir)
+{
+	int ret = 1;
+        char *orig, *s;
+	const char delim = '/';
+
+        if (!(orig = s = dbg_strdup((char*) dir)))
+		return log_alloc_err(lc, __func__);
+
+        /* Create parent directories */
+        log_notice(lc, "creating directory %s", dir);
+	do {
+		s = remove_delimiter(s + 1, delim);
+		if (mkdir(orig, 0777) && errno != EEXIST) {
+			log_err(lc, "mkdir %s", orig);
+			ret = 0;
+			break;
+		}
+
+		add_delimiter(&s, delim);
+	} while (s);
+
+	dbg_free(orig);
+
+        return ret;
+}
+
+/* Create directory. */
+int mk_dir(struct lib_context *lc, const char *dir)
+{
+	struct stat info;
+
+	/* If it doesn't exist yet, make it. */
+	if (stat(dir, &info))
+		return mk_dir_recursive(lc, dir);
+
+	if (S_ISDIR(info.st_mode))
+		return 1;
+
+	LOG_ERR(lc, 0, "directory %s not found", dir);
+}
+
+static int rw_file(struct lib_context *lc, const char *who, int flags,
+		   char *path, void *buffer, size_t size, loff_t offset)
+{
+	int fd, ret = 0;
+	loff_t o;
+	struct {
+		ssize_t (*func)();
+		const char *what;
+	} rw_spec[] = {
+		{ read,  "read" },
+		{ write, "writ" },
+	}, *rw = rw_spec + ((flags & O_WRONLY) ? 1 : 0);
+
+	if ((fd = open(path, flags, lc->mode)) == -1)
+		LOG_ERR(lc, 0, "opening \"%s\"", path);
+
+#ifdef __KLIBC__
+#define	DMRAID_LSEEK	lseek
+#else
+#define	DMRAID_LSEEK	lseek64
+#endif
+	if (offset && (o = DMRAID_LSEEK(fd, offset, SEEK_SET)) == (loff_t) -1)
+		log_err(lc, "%s: seeking device \"%s\" to %" PRIu64,
+			who, path, offset);
+	else if (rw->func(fd, buffer, size) != size)
+		log_err(lc, "%s: %sing %s[%s]", who, rw->what,
+			path, strerror(errno));
+	else
+		ret = 1;
+
+	close(fd);
+
+	return ret;
+}
+
+int read_file(struct lib_context *lc, const char *who, char *path,
+	      void *buffer, size_t size, loff_t offset)
+{
+	return rw_file(lc, who, O_RDONLY, path, buffer, size, offset);
+}
+
+int write_file(struct lib_context *lc, const char *who, char *path,
+	       void *buffer, size_t size, loff_t offset)
+{
+	/* O_CREAT|O_TRUNC are noops on a devnode. */
+	return rw_file(lc, who, O_WRONLY|O_CREAT|O_TRUNC, path,
+		       buffer, size, offset);
+}
/cvs/dm/dmraid/lib/misc/init.c,v  -->  standard output
revision 1.1
--- dmraid/lib/misc/init.c
+++ -	2008-02-22 16:57:43.738543000 +0000
@@ -0,0 +1,35 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include "internal.h"
+
+/* Library initialization. */
+struct lib_context *libdmraid_init(int argc, char **argv)
+{
+	struct lib_context *lc;
+
+	if ((lc = alloc_lib_context(argv))) {
+		if (!register_format_handlers(lc)) {
+			libdmraid_exit(lc);
+			lc = NULL;
+		} else
+			/* FIXME: do we need this forever ? */
+			sysfs_workaround(lc);
+	}
+
+	return lc;
+}
+
+/* Library exit processing. */
+void libdmraid_exit(struct lib_context *lc)
+{
+	free_raid_set(lc, NULL);	/* Free all RAID sets. */
+	free_raid_dev(lc, NULL);	/* Free all RAID devices. */
+	free_dev_info(lc, NULL);	/* Free all disk infos. */
+	unregister_format_handlers(lc);	/* Unregister all format handlers. */
+	free_lib_context(lc);		/* Release library context. */
+}
/cvs/dm/dmraid/lib/misc/lib_context.c,v  -->  standard output
revision 1.1
--- dmraid/lib/misc/lib_context.c
+++ -	2008-02-22 16:57:43.841458000 +0000
@@ -0,0 +1,189 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <stdarg.h>
+#include "internal.h"
+#include "version.h"
+
+/* Options access functions. */
+static inline int lc_opt_ok(enum lc_options o)
+{
+	return o < LC_OPTIONS_SIZE;
+}
+
+int lc_opt(struct lib_context *lc, enum lc_options o)
+{
+	return lc_opt_ok(o) ? lc->options[o].opt : 0;
+}
+
+static int _inc_opt(struct lib_context *lc, int o)
+{
+	return lc->options[o].opt < UCHAR_MAX ? ++lc->options[o].opt :
+					       lc->options[o].opt;
+}
+
+int lc_inc_opt(struct lib_context *lc, int o)
+{
+	return lc_opt_ok(o) ? _inc_opt(lc, o) : 0;
+}
+
+const char *lc_strcat_opt(struct lib_context *lc, enum lc_options o,
+			  char *arg, const char delim)
+{
+	char *ret = NULL;
+
+	if (lc_opt_ok(o)) {
+		char *a = (char*) OPT_STR(lc, o);
+		size_t end = (a ? strlen(a) : 0),
+		       len = end + strlen(arg) + ((delim && end) ? 1 : 0) + 1;
+
+		/* Dup new one. */
+		if ((ret = dbg_realloc(a, len))) {
+			if (delim && end)
+				ret[end++] = delim;
+
+			ret[end] = 0;
+			strcat(ret, arg);
+			OPT_STR(lc, o) = ret;
+		} else {
+			dbg_free((char*) OPT_STR(lc, o));
+			OPT_STR(lc, o) = ret;
+			log_alloc_err(lc, __func__);
+		}
+	}
+
+	return ret;
+}
+
+const char *lc_stralloc_opt(struct lib_context *lc, enum lc_options o,
+			    char *arg)
+{
+	if (lc_opt_ok(o)) {
+		/* Free any already allocated one. */
+		if (OPT_STR(lc, o))
+			dbg_free((char*) OPT_STR(lc, o));
+
+		/* Dup new one. */
+		if ((OPT_STR(lc, o) = dbg_strdup(arg)))
+			return OPT_STR(lc, o);
+
+		log_alloc_err(lc, __func__);
+	}
+
+	return NULL;
+}
+
+const char *lc_opt_arg(struct lib_context *lc, enum lc_options o)
+{
+	return lc_opt_ok(o) ? lc->options[o].arg.str : NULL;
+}
+
+struct list_head *lc_list(struct lib_context *lc, int l)
+{
+	return l < ARRAY_SIZE(lc->lists) ? lc->lists + l : NULL;
+}
+
+/*
+ * Library context initialization functions.
+ */
+static void init_options(struct lib_context *lc, void *arg)
+{
+	lc_inc_opt(lc, LC_SEPARATOR);
+	lc->options[LC_SEPARATOR].arg.str = dbg_strdup((char*) ",");
+}
+
+static void init_cmd(struct lib_context *lc, void *arg)
+{
+	lc->cmd = get_basename(lc, ((char**) arg)[0]);
+}
+
+static void init_lists(struct lib_context *lc, void *arg)
+{
+	unsigned int i = LC_LISTS_SIZE;
+
+	while (i--)
+		INIT_LIST_HEAD(lc->lists + i);
+}
+
+static void init_mode(struct lib_context *lc, void *arg)
+{
+	lc->mode = 0600;
+}
+
+static void init_paths(struct lib_context *lc, void *arg)
+{
+	lc->path.error = "/dev/zero";
+}
+
+/* FIXME: add lib flavour info (e.g., DEBUG). */
+static void init_version(struct lib_context *lc, void *arg)
+{
+	lc->version.text	= DMRAID_LIB_VERSION;
+	lc->version.date	= DMRAID_LIB_DATE;
+	lc->version.v.major	= DMRAID_LIB_MAJOR_VERSION;
+	lc->version.v.minor	= DMRAID_LIB_MINOR_VERSION;
+	lc->version.v.sub_minor	= DMRAID_LIB_SUBMINOR_VERSION;
+	lc->version.v.suffix	= DMRAID_LIB_VERSION_SUFFIX;
+}
+
+/* Put init functions into an array because of the potentially growing list. */
+struct init_fn {
+	void (*func)(struct lib_context *lc, void *arg);
+} init_fn[] = {
+	{ init_options },
+	{ init_cmd },
+	{ init_lists },
+	{ init_mode },
+	{ init_paths },
+	{ init_version },
+};
+
+struct lib_context *alloc_lib_context(char **argv)
+{
+	struct lib_context *lc;
+	struct init_fn *f;
+
+	if ((lc = dbg_malloc(sizeof(*lc)))) {
+		for (f = init_fn; f < ARRAY_END(init_fn); f++)
+			f->func(lc, argv);
+#ifdef	DEBUG_MALLOC
+		/*
+		 * Set DEBUG flag in case of memory debugging so that we
+		 * see messages even before the command line gets parsed.
+		 */
+		lc_inc_opt(lc, LC_DEBUG);
+#endif
+
+	} else
+		fprintf(stderr, "allocating library context\n");
+
+	return lc;
+}
+
+void free_lib_context(struct lib_context *lc)
+{
+	int o;
+
+	for (o = 0; o < LC_OPTIONS_SIZE; o++) {
+		if (lc->options[o].arg.str)
+			dbg_free ((char*) lc->options[o].arg.str);
+	}
+
+	dbg_free(lc);
+}
+
+/* Return library date (ASCII). */
+const char *libdmraid_date(struct lib_context *lc)
+{
+	return lc->version.date;
+}
+
+/* Return library version (ASCII). */
+const char *libdmraid_version(struct lib_context *lc)
+{
+	return lc->version.text;
+}
/cvs/dm/dmraid/lib/misc/misc.c,v  -->  standard output
revision 1.1
--- dmraid/lib/misc/misc.c
+++ -	2008-02-22 16:57:43.936402000 +0000
@@ -0,0 +1,224 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <stdarg.h>
+#include "internal.h"
+
+/* Prompt for a yes/no answer */
+int yes_no_prompt(struct lib_context *lc, const char *prompt, ...)
+{
+	int c = '\n';
+	va_list ap;
+
+	/* Use getc() for klibc compatibility. */
+	do {
+		if (c == '\n') {
+			va_start(ap, prompt);
+			vprintf(prompt, ap);
+			va_end(ap);
+			log_print_nnl(lc, " ? [y/n] :");
+		}
+	} while ((c = tolower(getc(stdin))) && c != 'y' && c != 'n');
+
+	/* Ignore rest. */
+	while (getc(stdin) != '\n');
+
+	return c == 'y';
+}
+
+/* Return the basename of a path. */
+char *get_basename(struct lib_context *lc, char *str)
+{
+	char *ret = strrchr(str, '/');
+
+	return ret ? ++ret : str;
+}
+
+/* Return the dirname of a path. */
+char *get_dirname(struct lib_context *lc, char *str)
+{
+	char *ret = strrchr(str, '/');
+	size_t len = ret ? ret - str : strlen(str);
+
+	if ((ret = dbg_malloc(len + 1)))
+		strncpy(ret, str, len);
+
+	return ret;
+}
+
+/* Convert a numeric string to alpha. */
+void mk_alpha(struct lib_context *lc, char *str, size_t len)
+{
+	for (; len && *str; len--, str++) {
+		if (isdigit(*str))
+			*str += 'a' - '0';
+	}
+}
+
+/* Remove any whitespace from a string. */
+char *remove_white_space(struct lib_context *lc, char *str, size_t size)
+{
+	int c;
+	char *in = str, *out = str;
+
+	in[size] = 0;
+	while ((c = *in++)) {
+		if (!isspace(c))
+			*out++ = c;
+	}
+	*out = 0;
+
+	return str;
+
+}
+
+/* Remove/add a delimiter character. */
+char *remove_delimiter(char *ptr, char c)
+{
+	char *ret = NULL;
+
+	if (ptr && (ret = strchr(ptr, (int) c)))
+		*ret = 0;
+
+	return ret; 
+}
+
+void add_delimiter(char **ptr, char c)
+{
+	if (ptr && *ptr) {
+		**ptr = c;
+		(*ptr)++;
+	}
+}
+
+/* Grow a string. */
+static int grow_string(struct lib_context *lc, char **string, const char *s)
+{
+	size_t len;
+	char *tmp = *string;
+
+ 	len = strlen(s) + (tmp ? strlen(tmp) + 1 : 1);
+	if ((*string = dbg_realloc(tmp, len))) {
+		if (!tmp)
+			**string = '\0';
+	} else if (tmp)
+		dbg_free(tmp);
+	
+	return *string ? 1 : 0;
+}
+
+/* Free a string. */
+void free_string(struct lib_context *lc, char **string)
+{
+	if (*string) {
+		dbg_free(*string);
+		*string = NULL;
+	}
+}
+
+/* Push a string onto the end of another. */
+static int p_str(struct lib_context *lc, char **string, const char *s)
+{
+	int ret;
+
+	if ((ret = grow_string(lc, string, s)))
+		strcat (*string, s);
+
+	return ret;
+}
+
+/* Push a string defined by a start and end pointer onto the end of another. */
+static int p_str_str(struct lib_context *lc, char **string,
+		     char *begin, char *end)
+{
+	if (end == begin)
+		return 1;
+
+	*end = 0;
+
+	return p_str(lc, string, begin);
+}
+
+/* Push an uint64_t in ascii onto the end of a string. */
+static int p_u64(struct lib_context *lc, char **string, const uint64_t u)
+{
+	char buffer[22];
+
+	sprintf(buffer, "%" PRIu64, u);
+
+	return p_str(lc, string, buffer);
+}
+
+/* Push an uint_t in ascii onto the end of a string. */
+static int p_u(struct lib_context *lc, char **string, const unsigned int u)
+{
+	return p_u64(lc, string, (uint64_t) u);
+}
+
+/* Push an uint_t in ascii onto the end of a string. */
+static int p_d(struct lib_context *lc, char **string, const int d)
+{
+	char buffer[12];
+
+	sprintf(buffer, "%d", d);
+
+	return p_str(lc, string, buffer);
+}
+
+/* Push a format string defined list of arguments onto a string. */
+int p_fmt(struct lib_context *lc, char **string, const char *fmt, ...)
+{
+	int ret = 1;
+	char *b, *f, *f_sav;
+	va_list ap;
+	
+	if (!(f = f_sav = dbg_strdup((char *) fmt)))
+		return 0;
+
+	va_start(ap, fmt);
+	while (ret && *(b = f++)) {
+		if (!(f = strchr(b, '%'))) {
+			/* No '%' -> just print string. */
+			ret = p_str(lc, string, b);
+			break;
+		}
+
+		if (!(ret = p_str_str(lc, string, b, f)))
+			break;
+
+		switch (*++f) {
+		case 'd':
+			ret = p_d(lc, string, va_arg(ap, int));
+			break;
+
+		case 's':
+			ret = p_str(lc, string, va_arg(ap, char *));
+			break;
+	
+		case 'u':
+			ret = p_u(lc, string, va_arg(ap, unsigned int));
+			break;
+
+		case 'U':
+			ret = p_u64(lc, string, va_arg(ap, uint64_t));
+			break;
+
+		default:
+			log_err(lc, "%s: unknown format identifier %%%c",
+				__func__, *f);
+			free_string(lc, string);
+			ret = 0;
+		}
+
+		f++;
+	}
+
+	va_end(ap);
+	dbg_free(f_sav);
+
+	return ret;
+}
/cvs/dm/dmraid/lib/misc/workaround.c,v  -->  standard output
revision 1.1
--- dmraid/lib/misc/workaround.c
+++ -	2008-02-22 16:57:44.048902000 +0000
@@ -0,0 +1,37 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <paths.h>
+#include "internal.h"
+
+/*
+ * OUCH (nasty hack).
+ *
+ * Need to open /dev/hd? in turn in order to
+ * populate /sys/block in case of IDE module
+ * load because of asynchronuous registration !?
+ */
+void sysfs_workaround(struct lib_context *lc)
+{
+	int d, fd;
+	size_t len;
+	char *dev;
+
+	if (!(dev = dbg_malloc(sizeof(_PATH_DEV) + 4)))
+		LOG_ERR(lc, , "sysfs workaround");
+
+	sprintf(dev, "%shd?", _PATH_DEV);
+	for (len = strlen(dev) - 1, d = 'a'; d <= 'z'; d++) {
+		dev[len] = (char) d;
+
+		if (!removable_device(lc, dev) &&
+		    (fd = open (dev, O_RDONLY)) != -1)
+			close (fd);
+	}
+
+	dbg_free(dev);
+}
/cvs/dm/dmraid/lib/mm/dbg_malloc.c,v  -->  standard output
revision 1.1
--- dmraid/lib/mm/dbg_malloc.c
+++ -	2008-02-22 16:57:44.139828000 +0000
@@ -0,0 +1,116 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#include <string.h>
+#include <stdlib.h>
+#include <sys/types.h>
+#include <stdint.h>
+#include "dbg_malloc.h"
+#include "log/log.h"
+
+static void *__dbg_malloc(size_t size, int init)
+{
+	void *ret = malloc(size);
+
+	if (init && ret)
+		memset(ret, 0, size);
+
+	return ret;
+}
+
+#ifdef	DEBUG_MALLOC
+
+void *_dbg_malloc(size_t size, struct lib_context *lc,
+		  const char *who, unsigned int line)
+{
+	void *ret = __dbg_malloc(size, 1);
+
+	log_dbg(lc, "%s: dbg_malloc(%zu) at line %u returned 0x%x",
+		(char*) who, size, line, (unsigned long) ret);
+
+	return ret;
+}
+
+void *_dbg_realloc(void *ptr, size_t size, struct lib_context *lc,
+		   const char *who, unsigned int line)
+{
+	void *ret = realloc(ptr, size);
+
+	log_dbg(lc, "%s: dbg_realloc(0x%x, %zu) at line %u returned 0x%x",
+		(char*) who, (unsigned long) ptr, size, line,
+		(unsigned long) ret);
+
+	return ret;
+}
+
+void *_dbg_strndup(void *ptr, size_t len, struct lib_context *lc,
+		   const char *who, unsigned int line)
+{
+	char *ret;
+
+	if ((ret = __dbg_malloc(len + 1, 0))) {
+		ret[len] = 0;
+		strncpy(ret, ptr, len);
+	}
+
+	log_dbg(lc, "%s: dbg_strndup(0x%x) at line %u returned 0x%x",
+		(char*) who, (unsigned long) ptr, line, (unsigned long) ret);
+
+	return ret;
+
+}
+
+void *_dbg_strdup(void *ptr, struct lib_context *lc,
+		  const char *who, unsigned int line)
+{
+	return _dbg_strndup(ptr, strlen(ptr), lc, who, line);
+}
+
+
+void _dbg_free(void *ptr, struct lib_context *lc,
+	       const char *who, unsigned int line)
+{
+	log_dbg(lc, "%s: dbg_free(0x%x) at line %u",
+		(char*) who, (unsigned long) ptr, line);
+	free(ptr);
+}
+
+#else
+
+void *_dbg_malloc(size_t size)
+{
+	return __dbg_malloc(size, 1);
+}
+
+void *_dbg_realloc(void *ptr, size_t size)
+{
+	return realloc(ptr, size);
+}
+
+void *_dbg_strndup(void *ptr, size_t len)
+{
+	char *ret;
+
+	if ((ret = __dbg_malloc(len + 1, 0))) {
+		ret[len] = 0;
+		strncpy(ret, ptr, len);
+	}
+
+	return ret;
+}
+
+void *_dbg_strdup(void *ptr)
+{
+	return _dbg_strndup(ptr, strlen(ptr));
+}
+
+void _dbg_free(void *ptr)
+{
+	free(ptr);
+}
+
+#endif /* #ifdef DEBUG_MALLOC */
/cvs/dm/dmraid/lib/mm/dbg_malloc.h,v  -->  standard output
revision 1.1
--- dmraid/lib/mm/dbg_malloc.h
+++ -	2008-02-22 16:57:44.228438000 +0000
@@ -0,0 +1,51 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _DBG_MALLOC_H_
+#define _DBG_MALLOC_H_
+
+#include <stdio.h>
+#include <sys/types.h>
+
+#ifdef	DEBUG_MALLOC
+
+struct lib_context;
+void *_dbg_malloc(size_t size, struct lib_context *lc,
+		  const char *who, unsigned int line);
+void *_dbg_realloc(void *ptr, size_t size, struct lib_context *lc,
+		   const char *who, unsigned int line);
+void *_dbg_strdup(void *ptr, struct lib_context *lc,
+		  const char *who, unsigned int line);
+void *_dbg_strndup(void *ptr, size_t len, struct lib_context *lc,
+		   const char *who, unsigned int line);
+void _dbg_free(void *ptr, struct lib_context *lc,
+	       const char *who, unsigned int line);
+
+#define	dbg_malloc(size)	_dbg_malloc((size), lc, __func__, __LINE__)
+#define	dbg_realloc(ptr, size)	_dbg_realloc((ptr), (size), lc, \
+					     __func__, __LINE__)
+#define	dbg_strdup(ptr)		_dbg_strdup((ptr), lc, __func__, __LINE__)
+#define	dbg_strndup(ptr, len)	_dbg_strdup((ptr), len, lc, __func__, __LINE__)
+#define	dbg_free(ptr)		_dbg_free((ptr), lc, __func__, __LINE__)
+
+#else
+
+void *_dbg_malloc(size_t size);
+void *_dbg_realloc(void *ptr, size_t size);
+void *_dbg_strdup(void *ptr);
+void *_dbg_strndup(void *ptr, size_t len);
+void _dbg_free(void *ptr);
+
+#define	dbg_malloc	_dbg_malloc
+#define	dbg_realloc	_dbg_realloc
+#define	dbg_strdup	_dbg_strdup
+#define	dbg_strndup	_dbg_strndup
+#define	dbg_free	_dbg_free
+
+#endif /* #ifdef DEBUG_MALLOC */
+
+#endif
/cvs/dm/dmraid/man/Makefile.in,v  -->  standard output
revision 1.1
--- dmraid/man/Makefile.in
+++ -	2008-02-22 16:57:44.336224000 +0000
@@ -0,0 +1,36 @@
+#
+# Copyright (C) 2004-2005  Heinz Mauelshagen, Red Hat GmbH. All rights reserved.
+#
+# See file LICENSE at the top of this source tree for license information.
+#
+
+srcdir = @srcdir@
+top_srcdir = @top_srcdir@
+VPATH = @srcdir@
+
+MAN8=dmraid.8
+MAN8DIR=${mandir}/man8
+
+include $(top_srcdir)/make.tmpl
+
+.PHONY:	install_dmraid_man remove_dmraid_man
+
+install_dmraid_man:
+	@echo "Installing $(MAN8) in $(MAN8DIR)"; \
+	mkdir -p $(MAN8DIR); \
+	for f in $(MAN8); \
+	do \
+	   $(RM) $(MAN8DIR)/$$f; \
+	   @INSTALL@ -D -m 444 $$f $(MAN8DIR)/$$f; \
+	done
+
+install:	install_dmraid_man
+
+remove_dmraid_man:
+	@echo "Removing $(MAN8) in $(MAN8DIR)"; \
+	for f in $(MAN8); \
+	do \
+	   $(RM) $(MAN8DIR)/$$f; \
+	done
+
+remove:	remove_dmraid_man
/cvs/dm/dmraid/man/dmraid.8,v  -->  standard output
revision 1.1
--- dmraid/man/dmraid.8
+++ -	2008-02-22 16:57:44.506125000 +0000
@@ -0,0 +1,414 @@
+.TH DMRAID 8 "DMRAID TOOL" "Heinz Mauelshagen" \" -*- nroff -*-
+.SH NAME
+dmraid \- discover and activate software (ATA)RAID
+.SH SYNOPSIS
+.B dmraid
+ {-a|--activate} {y|n|yes|no} 
+ [-d|--debug]... [-v|--verbose]... [-i|--ignorelocking]
+ [-f|--format FORMAT[,FORMAT...]]
+ [-p|--no_partitions]
+ [--separator SEPARATOR]
+ [-t|--test]
+ [RAID-set...]
+
+.B dmraid
+ {-b|--block_devices}
+ [-c|--display_columns][FIELD[,FIELD...]]...
+ [-d|--debug]... [-v|--verbose]...
+ [--separator SEPARATOR]
+ [device-path...]
+
+.B dmraid
+ {-h|--help}
+
+.B dmraid
+ {-l|--list_formats}
+ [-d|--debug]... [-v|--verbose]...
+
+.B dmraid
+ {-n|--native_log}
+ [-d|--debug]... [-v|--verbose]... [-i|--ignorelocking]
+ [-f|--format FORMAT[,FORMAT...]]
+ [--separator SEPARATOR]
+ [device-path...]
+
+.B dmraid
+ {-r|--raid_devices}
+ [-c|--display_columns][FIELD[,FIELD...]]...
+ [-d|--debug]... [-v|--verbose]... [-i|--ignorelocking]
+ [-D|--dump_metadata]
+ [-f|--format FORMAT[,FORMAT...]]
+ [--separator SEPARATOR]
+ [device-path...]
+
+.B dmraid
+ {-r|--raid_devices}
+ [-d|--debug]... [-v|--verbose]... [-i|--ignorelocking]
+ [-E|--erase_metadata]
+ [-f|--format FORMAT[,FORMAT...]]
+ [--separator SEPARATOR]
+ [device-path...]
+
+.B dmraid
+ {-s|--sets}...[a|i|active|inactive]
+ [-c|--display_columns][FIELD[,FIELD...]]...
+ [-d|--debug]... [-v|--verbose]... [-i|--ignorelocking]
+ [-f|--format FORMAT[,FORMAT...]]
+ [-g|--display_group]
+ [--separator SEPARATOR]
+ [RAID-set...]
+
+.B dmraid
+ {-V/--version}
+
+.SH DESCRIPTION
+dmraid discovers block and software RAID devices (eg, ATARAID)
+by using multiple different metadata format handlers which
+support various formats (eg, Highpoint 37x series).
+It offers activating RAID sets made up by 2 or more
+discovered RAID devices, display properties of devices and sets (see option
+.B -l
+for supported metadata formats).
+Block device access to activated RAID sets occurs via device-mapper nodes
+/dev/mapper/RaidSetName.
+RaidSetName starts with the format name (see
+.B -l
+option) which can be used to access all RAID sets of a specific format
+easily with certain options (eg,
+.B -a
+below).
+
+.SS OPTIONS
+.TP
+.I \-a, \-\-activate {y|n} [RAID set...]
+Activates or deactivates all or particular software RAID set.
+In case metadata format handlers are chosen with
+.B -f
+, only RAID sets with such format(s) can be activated or deactivated.
+Useful if devices have multiple metadata signatures.
+When activating RAID sets,
+.B -p
+disables the activation of partitions on them.
+RAID set names given on command line don't need to be fully specified
+(eg, "dmraid -ay sil" would activate all discovered Silicon Image Medley
+RAID sets).
+
+.TP
+.I {-b|--block_devices} [device-path...]
+List all or particular discovered block devices with their
+properties (size, serial number).
+Add
+.B -c
+to display block device names only and
+.B -cc
+for CSV column output of block device properties.
+See description of
+.B -c
+below for FIELD identifiers.
+
+.TP
+.I [-d|--debug]...
+Enable debugging output. Opion can be given multiple times
+increasing the debug output level.
+
+.TP
+.I [-c|--display_columns][FIELD[,FIELD...]]...
+Display properties of block devices, RAID sets and devices in column(s).
+Optional list specifying which FIELDs to display.
+.br
+For
+.B -b:
+.br
+d[evpath]|p[ath], sec[tors]|si[ze], ser[ialnumber].
+.br
+For
+.B -r:
+.br
+de[vpath]|p[ath], f[ormat], r[aidname], t[ype], st[atus], se[ctors]|si[ze], da[taoffset]|o[ffset].
+.br
+For
+.B -s:
+.br
+f[ormat], r[aidname], t[ype], sta[tus], str[ide], se[ctors]|si[ze], su[bsets], d[evices], sp[ares].
+.br
+.TP
+.I [-f|--format FORMAT[,FORMAT...]]
+Use metadata format handler(s) to discover RAID devices.
+See
+.B -l
+for a list of supported format handler names. This is useful to
+select particular formats in case multiple metadata signatures are found
+on a device. A comma seperated list of format names can be specified which 
+may not contain white space.
+
+.TP
+.I {-h|--help}
+Display help text.
+
+.TP
+.I {-i|--ignorelocking}
+Don't take out any locks. Useful in early boot where no read/write
+access to /var is available.
+
+.TP
+.I {-l|--list_formats}
+List all available metadata format handlers with their names and
+descriptions. Supported RAID levels are listed in parenthesis:
+.br
+
+S: Span (concatination)
+.br
+0: RAID0 (stripe)
+.br
+1: RAID1 (mirror)
+.br
+10: RAID10 (mirror on top of stripes)
+.br
+01: RAID10 (stripe on top of mirrors)
+
+.TP
+.I {-n|--native_log} [device-path...]
+Display metadata in native, vendor-specific format.
+In case a metadata format handler is chosen with
+.B -f
+only RAID devices with such format will be displayed in native format.
+If device-path(s) is/are given on the command line, native metadata output
+is restricted to those listed.
+
+.TP
+.I {-r|--raid_devices} [device-path...]
+List all discovered RAID devices with format, RAID level, sectors used
+and data offset into the device.
+In case a metadata format handler is chosen with
+.B -f
+, only RAID devices with such format can be discovered. Useful if devices
+have multiple metadata signatures.
+If 
+.B -D
+is added to
+.B -r
+the RAID metadata gets dumped into files named devicename_formatname.dat.
+The offset in sectors where the metadata is located on the device is written
+into files named devicename_formatname.offset and the size of the device
+into files named devicename_formatname.size.
+
+If 
+.B -E
+is added to
+.B -r
+the RAID metadata on the devices gets conditionally erased.
+Useful to erase old metadata after new one of different type has been
+stored on a device in order to avoid discovering both. If you enter
+.B -E
+option
+.B -D
+will be enforced in order to have a fallback in case the wrong metadata
+got erased.
+Manual copying back onto the device is needed to recover from erasing
+the wrong metadata using the dumped files devicename_formatname.dat
+and devicename_formatname.offset.
+Eg, to restore all *.dat files in the working directory to the respective devices:
+
+.br
+for f in *.dat
+.br
+do
+.br
+	dd if=$f of=/dev/${f%%_*} \\
+.br
+	seek=`cat ${f%%dat}offset` bs=1
+.br
+done
+.br
+
+If device-path(s) is/are given on the command line, the above actions
+are restricted to those listed.
+Add
+.B -c
+to display RAID device names only and
+.B -cc
+for CSV column output of RAID device properties.
+See description of
+.B -c
+above for FIELD identifiers.
+
+.TP
+.I --separator SEPARATOR
+Use SEPARATOR as a delimiter for all options taking or displaying lists.
+
+.TP
+.I -s... [a|i] [RAID-set...]
+Display properties of RAID sets. Multiple RAID set names can be given
+on the command line which don't need to be fully specified (eg, "dmraid -s hpt"
+would display all discovered Highpoint RAID sets). Enter
+.B -s
+twice to display RAID subsets too.
+Add
+.B -c
+to display names of RAID sets only,
+.B -cc
+for CSV column output of RAID set properties and
+.B -ccc
+for inclusion of block devices in the listing. Doesn't imply
+.B -s -s
+to show RAID subsets (implied for group sets, e.g. isw).
+Add
+.B -g
+to include information about group RAID sets (as with Intel Software
+RAID) in the listing.
+See description of
+.B -c
+above for FIELD identifiers.
+
+.TP
+.I [-v|--verbose]...
+Enable verbose runtime information output. Opion can be given multiple times
+increasing the verbosity level.
+
+.SH EXAMPLES
+"dmraid -l" lists all supported metadata formats with their names along with
+some descriptive information, eg:
+.br
+hpt37x : (+) Highpoint HPT37X
+.br
+hpt45x : (+) Highpoint HPT45X
+.br
+isw    : (+) Intel Software RAID
+.br
+lsi    : (0) LSI Logic MegaRAID
+.br
+nvidia : (+) NVidia RAID
+.br
+pdc    : (+) Promise FastTrack
+.br
+sil    : (+) Silicon Image(tm) Medley(tm)
+.br
+via    : (+) VIA Software RAID
+.br
+dos    : (+) DOS partitions on SW RAIDs
+.br
+(0): Discover, (+): Discover+Activate
+
+"dmraid -ay" activates all software RAID sets discovered.
+
+"dmraid -an" deactivates all active software RAID sets which are not open
+(eg, mounted filesystem on them).
+
+"dmraid -ay -f pdc" (pdc looked up from "dmraid -l") activates all
+software RAID sets with Promise format discovered and ignores all other
+supported formats.
+
+"dmraid -r" discovers all software RAID devices supported on your system, eg:
+.br
+/dev/dm-46: hpt45x, "hpt45x_chidjhaiaa-0", striped, ok, 320172928 sectors, data@ 0
+.br
+/dev/dm-50: hpt45x, "hpt45x_chidjhaiaa-0", striped, ok, 320172928 sectors, data@ 0
+.br
+/dev/dm-54: hpt45x, "hpt45x_chidjhaiaa-1", striped, ok, 320172928 sectors, data@ 0
+.br
+/dev/dm-58: hpt45x, "hpt45x_chidjhaiaa-1", striped, ok, 320172928 sectors, data@ 0
+
+
+"dmraid -s -s hpt45x_chidjhaiaa" displays properties of
+set "hpt45x_chidjhaiaa", eg:
+.br
+*** Superset
+.br
+name   : hpt45x_chidjhaiaa
+.br
+size   : 640345856
+.br
+stride : 128
+.br
+type   : raid10
+.br
+status : ok
+.br
+subsets: 2
+.br
+dev    : 4
+.br
+spare  : 0
+.br
+---> Subset
+.br
+name   : hpt45x_chidjhaiaa-0
+.br
+size   : 640345856
+.br
+stride : 128
+.br
+type   : stripe
+.br
+status : ok
+.br
+subsets: 0
+.br
+dev    : 2
+.br
+spare  : 0
+.br
+---> Subset
+.br
+name   : hpt45x_chidjhaiaa-1
+.br
+size   : 640345856
+.br
+stride : 128
+.br
+type   : stripe
+.br
+status : ok
+.br
+subsets: 0
+.br
+dev    : 2
+.br
+spare  : 0
+.br
+
+"dmraid -s -ccs hpt45" displays properties in column format of all sets
+and subsets with hpt45* format, eg:
+.br
+hpt45x_chidjhaiaa,640345856,128,raid10,ok,4,0
+.br
+hpt45x_chidjhaiaa-a,640345856,128,stripe,ok,2,0
+.br
+hpt45x_chidjhaiaa-b,640345856,128,stripe,ok,2,0
+
+"dmraid -r --sep : -cpath:size" display paths and sizes in sectors for
+RAID devices in column format using ':' as a delimiter, eg:
+.br
+/dev/dm-8:320173055
+.br
+/dev/dm-12:320173055
+.br
+/dev/dm-22:320173055
+.br
+/dev/dm-26:320173055
+.br
+/dev/dm-30:586114703
+.br
+/dev/dm-34:586114703
+.br
+/dev/dm-38:586114703
+.br
+/dev/dm-42:586114703
+.br
+/dev/dm-46:156301487
+.br
+/dev/dm-50:156301487
+.br
+/dev/dm-54:390624896
+.br
+/dev/dm-58:390624896
+.br
+/dev/dm-62:390624896
+.br
+/dev/dm-66:390624896
+
+.SH DIAGNOSTICS
+dmraid returns an exit code of 0 for success or 1 for error.
+
+.SH AUTHOR
+Heinz Mauelshagen <Mauelshagen at RedHat.com>
/cvs/dm/dmraid/tools/Makefile.in,v  -->  standard output
revision 1.1
--- dmraid/tools/Makefile.in
+++ -	2008-02-22 16:57:45.039868000 +0000
@@ -0,0 +1,75 @@
+#
+# Copyright (C) 2004-2005  Heinz Mauelshagen, Red Hat GmbH. All rights reserved.
+#
+# See file LICENSE at the top of this source tree for license information.
+#
+
+srcdir = @srcdir@
+top_srcdir = @top_srcdir@
+VPATH = @srcdir@
+DIETLIBC = @DIETLIBC@
+KIBC = @KLIBC@
+
+INCLUDES += -I $(src_dir)
+
+SOURCES=\
+	dmraid.c \
+	commands.c \
+	toollib.c
+
+TARGETS=\
+	dmraid
+
+DMRAIDLIBS=-ldmraid
+
+include $(top_srcdir)/make.tmpl
+
+ifeq ("@KLIBC@", "yes")
+	DMRAIDLIBS += -ldevmapper_klibc
+else
+	ifeq ("@DIETLIBC@", "yes")
+		DMRAIDLIBS += -ldevmapper_dietc
+	else
+		DMRAIDLIBS += -ldevmapper
+	endif
+
+	ifeq ("@LIBSELINUX@", "yes")
+		DMRAIDLIBS += -lselinux
+	endif
+
+	ifeq ("@LIBSEPOL@", "yes")
+		DMRAIDLIBS += -lsepol
+	endif
+endif
+
+# We just link klibc static and get it over with
+ifeq ("@KLIBC@", "no")
+	ifeq ("@STATIC_LINK@", "no")
+		LDFLAGS += -rdynamic
+	else
+		LDFLAGS += -static
+	endif
+endif
+
+
+.PHONY: install_dmraid_tools remove_dmraid_tools
+
+dmraid:	$(OBJECTS) $(top_srcdir)/lib/libdmraid.a
+	$(CC) -o $@ $(OBJECTS) $(LDFLAGS) -L$(top_srcdir)/lib \
+	      -L$(DESTDIR)/lib $(DMRAIDLIBS) $(LIBS)
+
+install_dmraid_tools: $(TARGETS)
+	@echo "Installing $(TARGETS) in $(sbindir)"; \
+	mkdir -p $(sbindir); \
+	$(INSTALL) $(STRIP) $(TARGETS) $(sbindir)
+
+install: install_dmraid_tools
+
+remove_dmraid_tools:
+	@echo "Removing $(TARGETS) from $(sbindir)"; \
+	for f in $(TARGETS); \
+	do \
+		rm -f $(sbindir)/$$f; \
+	done
+
+remove:	remove_dmraid_tools
/cvs/dm/dmraid/tools/VERSION,v  -->  standard output
revision 1.1
--- dmraid/tools/VERSION
+++ -	2008-02-22 16:57:45.162011000 +0000
@@ -0,0 +1 @@
+1.0.0.rc11 (2006.05.15)
/cvs/dm/dmraid/tools/commands.c,v  -->  standard output
revision 1.1
--- dmraid/tools/commands.c
+++ -	2008-02-22 16:57:45.303788000 +0000
@@ -0,0 +1,791 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifdef HAVE_GETOPTLONG
+# define _GNU_SOURCE
+# include <getopt.h>
+#endif
+
+#include <string.h>
+#include <unistd.h>
+#include <dmraid/dmraid.h>
+#include "../lib/log/log.h"
+#include "commands.h"
+#include "toollib.h"
+#include "version.h"
+
+/* Action flags */
+enum action action = UNDEF;
+
+/*
+ * Command line options.
+ */
+static char const *short_opts = "a:hip"
+#ifndef	DMRAID_MINI
+				"bc::dDEf:gl"
+#ifdef	DMRAID_NATIVE_LOG
+				"n"
+#endif
+				"rs::tv"
+#endif
+				"V";
+
+#ifdef HAVE_GETOPTLONG
+static struct option long_opts[] = {
+	{"activate", required_argument, NULL, 'a'},
+	{"format", required_argument, NULL, 'f'},
+	{"no_partitions", no_argument, NULL, 'p'},
+# ifndef DMRAID_MINI
+	{"block_devices", no_argument, NULL, 'b'},
+	{"display_columns", optional_argument, NULL, 'c'},
+	{"debug", no_argument, NULL, 'd'},
+	{"dump_metadata", no_argument, NULL, 'D'},
+	{"erase_metadata", no_argument, NULL, 'E'},
+	{"display_group", no_argument, NULL, 'g'},
+# endif
+	{"help", no_argument, NULL, 'h'},
+	{"ignorelocking", no_argument, NULL, 'i'},
+# ifndef DMRAID_MINI
+	{"list_formats", no_argument, NULL, 'l'},
+#  ifdef DMRAID_NATIVE_LOG
+	{"native_log", no_argument, NULL, 'n'},
+#  endif
+	{"raid_devices", no_argument, NULL, 'r'},
+	{"sets", optional_argument, NULL, 's'},
+	{"separator", required_argument, NULL, SEPARATOR}, /* long only. */
+	{"test", no_argument, NULL, 't'},
+	{"verbose", no_argument, NULL, 'v'},
+# endif
+	{"version", no_argument, NULL, 'V'},
+	{NULL, no_argument, NULL, 0}
+};
+#endif /* #ifdef HAVE_GETOPTLONG */
+
+/* Definitions of option strings and actions for check_optarg(). */
+struct optarg_def {
+	const char *str;
+	const enum action action;
+};
+
+/* Check option argument. */
+static int check_optarg(struct lib_context *lc, const char option,
+			struct optarg_def *def)
+{
+	size_t len;
+	struct optarg_def *d;
+
+	if (optarg)
+		str_tolower(optarg);
+	else
+		return 1;
+
+	for (d = def, len = strlen(optarg); d->str; d++) {
+		if (!strncmp(optarg, d->str, len)) {
+			action |= d->action;
+			return 1;
+		}
+	}
+
+	LOG_ERR(lc, 0, "Invalid option argument for -%c", option);
+}
+
+/* Check activate/deactivate option arguments. */
+static int check_activate(struct lib_context *lc, int arg)
+{
+	struct optarg_def def[] = {
+		{ "yes", ACTIVATE },
+		{ "no",  DEACTIVATE },
+		{ NULL,  UNDEF },
+	};
+
+	return check_optarg(lc, 'a', def);
+}
+
+#ifndef	DMRAID_MINI
+/* Check active/inactive option arguments. */
+static int check_active(struct lib_context *lc, int arg)
+{
+	struct optarg_def def[] = {
+		{ "active",   ACTIVE },
+		{ "inactive", INACTIVE },
+		{ NULL,  UNDEF },
+	};
+
+	lc_inc_opt(lc, LC_SETS);
+
+	return check_optarg(lc, 's', def);
+}
+
+/* Check and store option arguments. */
+static int check_identifiers(struct lib_context *lc, int o)
+{
+	if (optarg) {
+		const char delim = *OPT_STR_SEPARATOR(lc);
+		char *p = optarg;
+
+		p = remove_white_space(lc, p, strlen(p));
+		p = collapse_delimiter(lc, p, strlen(p), delim);
+		if (!lc_strcat_opt(lc, o, p, delim))
+			return 0;
+	}
+
+	lc_inc_opt(lc, o);
+
+	return 1;
+}
+
+/* Check and store option argument/output field separator. */
+static int check_separator(struct lib_context *lc, int arg)
+{
+	if (strlen(optarg) != 1)
+		LOG_ERR(lc, 0, "Invalid separator \"%s\"", optarg);
+
+	return lc_stralloc_opt(lc, LC_SEPARATOR, optarg) ? 1 : 0;
+}
+#endif
+
+/* Display help information */
+static int help(struct lib_context *lc, int arg)
+{
+	char *c = lc->cmd;
+
+#ifdef	DMRAID_MINI
+	log_print(lc, "%s: Device-Mapper Software RAID tool "
+		  "[Early Boot Version]\n", c);
+	log_print(lc, "%s\t{-a|--activate} {y|n|yes|no} [-i|--ignorelocking]\n"
+		  "\t[-f|--format FORMAT[,FORMAT...]]\n"
+		  "\t[-p|--no_partitions]\n"
+		  "\t[--separator SEPARATOR]\n"
+		  "\t[RAID-set...]\n", c);
+	log_print(lc, "%s\t{-h|--help}\n", c);
+	log_print(lc, "%s\t{-V/--version}\n", c);
+#else
+	log_print(lc, "%s: Device-Mapper Software RAID tool\n", c);
+	log_print(lc, "* = [-d|--debug]... [-v|--verbose]... [-i|--ignorelocking]\n");
+	log_print(lc, "%s\t{-a|--activate} {y|n|yes|no} *\n"
+		  "\t[-f|--format FORMAT[,FORMAT...]]\n"
+		  "\t[-p|--no_partitions]\n"
+		  "\t[--separator SEPARATOR]\n"
+		  "\t[-t|--test]\n"
+		  "\t[RAID-set...]\n", c);
+	log_print(lc, "%s\t{-b|--block_devices} *\n"
+		  "\t[-c|--display_columns][FIELD[,FIELD...]]...\n"
+		  "\t[device-path...]\n", c);
+	log_print(lc, "%s\t{-h|--help}\n", c);
+	log_print(lc, "%s\t{-l|--list_formats} *\n", c);
+#  ifdef	DMRAID_NATIVE_LOG
+	log_print(lc, "%s\t{-n|--native_log} *\n"
+		  "\t[-f|--format FORMAT[,FORMAT...]]\n"
+		  "\t[--separator SEPARATOR]\n"
+		  "\t[device-path...]\n", c);
+#  endif
+	log_print(lc, "%s\t{-r|--raid_devices} *\n"
+		  "\t[-c|--display_columns][FIELD[,FIELD...]]...\n"
+		  "\t[-D|--dump_metadata]\n"
+		  "\t[-f|--format FORMAT[,FORMAT...]]\n"
+		  "\t[--separator SEPARATOR]\n"
+		  "\t[device-path...]\n", c);
+	log_print(lc, "%s\t{-r|--raid_devices} *\n"
+		  "\t{-E|--erase_metadata}\n"
+		  "\t[-f|--format FORMAT[,FORMAT...]]\n"
+		  "\t[--separator SEPARATOR]\n"
+		  "\t[device-path...]\n", c);
+	log_print(lc, "%s\t{-s|--sets}...[a|i|active|inactive] *\n"
+		  "\t[-c|--display_columns][FIELD[,FIELD...]]...\n"
+		  "\t[-f|--format FORMAT[,FORMAT...]]\n"
+		  "\t[-g|--display_group]\n"
+		  "\t[--separator SEPARATOR]\n"
+		  "\t[RAID-set...]\n", c);
+	log_print(lc, "%s\t{-V/--version}\n", c);
+#endif
+
+	return 1;
+}
+
+/*
+ * Action flag definitions for set_action()
+ *
+ * 'Early' options can be handled directly in set_action() by calling
+ * the functions registered here (set on_set member).
+ */
+static struct actions actions[] = {
+	/* [De]activate option. */
+	{ 'a',
+	  UNDEF, /* Set in check_activate() by mandatory option argument. */
+	  UNDEF,
+	  ACTIVATE|DEACTIVATE|FORMAT|HELP|IGNORELOCKING|NOPARTITIONS|SEPARATOR
+#ifndef DMRAID_MINI
+	  |DBG|TEST|VERBOSE
+#endif
+ 	  , ARGS,
+	  check_activate,
+	  0,
+	},
+
+	/* Format option. */
+	{ 'f',
+	  FORMAT,
+	  ACTIVATE|DEACTIVATE
+#ifndef DMRAID_MINI
+#  ifdef DMRAID_NATIVE_LOG
+	  |NATIVE_LOG
+#  endif
+	  |RAID_DEVICES|RAID_SETS,
+	  ACTIVE|INACTIVE|COLUMN|DBG|DUMP|ERASE|GROUP|HELP|
+	  IGNORELOCKING|NOPARTITIONS|SEPARATOR|TEST|VERBOSE
+#else
+	  , UNDEF
+#endif
+	  , ARGS,
+#ifndef DMRAID_MINI
+	  check_identifiers,
+#else
+	  NULL,
+#endif
+	  LC_FORMAT,
+	},
+
+	/* Partition option. */
+	{ 'p',
+	  NOPARTITIONS,
+	  ACTIVATE|DEACTIVATE,
+	  FORMAT|HELP|IGNORELOCKING|SEPARATOR
+#ifndef DMRAID_MINI
+	  |DBG|TEST|VERBOSE
+#endif
+	  , ARGS,
+	  NULL,
+	  0,
+	},
+
+#ifndef	DMRAID_MINI
+	/* Block devices option. */
+	{ 'b',
+	  BLOCK_DEVICES,
+	  UNDEF,
+	  COLUMN|DBG|HELP|IGNORELOCKING|SEPARATOR|VERBOSE,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_DEVICES,
+	},
+
+	/* Columns display option. */
+	{ 'c',
+	  COLUMN,
+	  BLOCK_DEVICES|RAID_DEVICES|RAID_SETS,
+	  ACTIVE|INACTIVE|DBG|DUMP|FORMAT|GROUP|HELP|IGNORELOCKING
+	  |SEPARATOR|VERBOSE,
+	  ARGS,
+	  check_identifiers,
+	  LC_COLUMN,
+	},
+
+	/* Debug option. */
+	{ 'd',
+	  DBG,
+	  ALL_FLAGS,
+	  ALL_FLAGS,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_DEBUG,
+	},
+
+	/* Dump metadata option. */
+	{ 'D',
+	  DUMP,
+	  RAID_DEVICES,
+	  COLUMN|DBG|FORMAT|HELP|IGNORELOCKING|SEPARATOR|VERBOSE,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_DUMP,
+	},
+
+	/* Erase metadata option. */
+	{ 'E',
+	  ERASE,
+	  RAID_DEVICES,
+	  COLUMN|DBG|FORMAT|HELP|IGNORELOCKING|SEPARATOR|VERBOSE,
+	  ARGS,
+	  NULL,
+	  0,
+	},
+
+	/* RAID groups option. */
+	{ 'g',
+	  GROUP,
+	  RAID_SETS,
+	  ACTIVE|INACTIVE|DBG|COLUMN|FORMAT|HELP|IGNORELOCKING
+	  |SEPARATOR|VERBOSE,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_GROUP,
+	},
+
+#endif
+	/* Help option. */
+	{ 'h',
+	  HELP,
+	  UNDEF,
+	  ALL_FLAGS,
+	  ARGS,
+	  help,
+	  0, 
+	},
+
+	/* ignorelocking option. */
+	{ 'i',
+	  IGNORELOCKING,
+	  UNDEF,
+	  ALL_FLAGS,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_IGNORELOCKING,
+	},
+
+#ifndef	DMRAID_MINI
+	/* List metadata format handlers option. */
+	{ 'l',
+	  LIST_FORMATS,
+	  UNDEF,
+	  DBG|HELP|IGNORELOCKING|VERBOSE,
+	  NO_ARGS,
+	  NULL,
+	  0,
+	},
+
+#  ifdef DMRAID_NATIVE_LOG
+	/* Native log option. */
+	{ 'n',
+	  NATIVE_LOG,
+	  UNDEF,
+	  DBG|FORMAT|HELP|IGNORELOCKING|SEPARATOR|VERBOSE,
+	  ARGS,
+	  NULL,
+	  0,
+	},
+
+#  endif
+	/* Display RAID devices option. */
+	{ 'r',
+	  RAID_DEVICES,
+	  UNDEF,
+	  COLUMN|DBG|DUMP|ERASE|FORMAT|HELP|IGNORELOCKING|SEPARATOR|VERBOSE,
+	  ARGS,
+	  NULL,
+	  0,
+	},
+
+	/* Display RAID sets option. */
+	{ 's',
+	  RAID_SETS,
+	  UNDEF,
+	  ACTIVE|INACTIVE|COLUMN|DBG|FORMAT|GROUP|HELP|IGNORELOCKING
+	  |SEPARATOR|VERBOSE,
+	  ARGS,
+	  check_active,
+	  0,
+	},
+
+	/* Display RAID sets option. */
+	{ SEPARATOR,
+	  SEPARATOR,
+	  COLUMN|FORMAT,
+	  ALL_FLAGS,
+	  ARGS,
+	  check_separator,
+	  0,
+	},
+
+
+	/* Test run option. */
+	{ 't',
+	  TEST,
+	  ACTIVATE|DEACTIVATE,
+	  ACTIVATE|DEACTIVATE|DBG|FORMAT|HELP|IGNORELOCKING|
+	  NOPARTITIONS|VERBOSE,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_TEST,
+	},
+
+	/* Verbose option. */
+	{ 'v',
+	  VERBOSE,
+	  ALL_FLAGS,
+	  ALL_FLAGS,
+	  ARGS,
+	  lc_inc_opt,
+	  LC_VERBOSE,
+	},
+#endif /* #ifndef DMRAID_MINI */
+
+	/* Version option. */
+	{ 'V',
+	  VERSION,
+	  UNDEF,
+#ifdef DMRAID_MINI
+	  HELP,IGNORELOCKING,
+#else
+	  DBG|HELP|IGNORELOCKING|VERBOSE,
+#endif
+	  NO_ARGS,
+	  NULL,
+	  0,
+	},
+};
+
+/*
+ * Set action flag and call optional function.
+ */
+static int set_action(struct lib_context *lc, int o)
+{
+	struct actions *a;
+
+	for (a = actions; a < ARRAY_END(actions); a++) {
+		if (o == a->option) {
+			action |= a->action;	/* Set action flag. */
+			a->allowed |= a->action;/* Merge to allowed flags. */
+			a->allowed |= a->needed;
+			if (a->f_set)	/* Optionally call function. */
+				return a->f_set(lc, a->arg);
+
+			break;
+		}
+	}
+
+	return 1;
+}
+
+/* Check for invalid option combinations */
+static int check_actions(struct lib_context *lc, char **argv)
+{
+	struct actions *a;
+
+	for (a = actions; a < ARRAY_END(actions); a++) {
+		if (a->action & action) {
+			if (a->needed != UNDEF &&
+			    !(a->needed & action))
+				LOG_ERR(lc, 0,
+					"option missing/invalid option "
+					"combination with -%c",
+					a->option);
+
+			if (~a->allowed & action)
+				LOG_ERR(lc, 0, "Invalid option combination"
+					   " (-h for help)");
+
+			if (a->args == NO_ARGS && argv[optind])
+				LOG_ERR(lc, 0,
+					"No arguments allowed with -%c\n",
+					a->option);
+		}
+	}
+
+	if (!action)
+		LOG_ERR(lc, 0, "Options missing\n");
+
+#ifndef DMRAID_MINI
+	if ((action & (DBG|VERBOSE)) == action)
+		LOG_ERR(lc, 0, "More options needed with -d/-v");
+
+	if (action & ERASE) {
+		action |= DUMP;
+		lc_inc_opt(lc, LC_DUMP);
+	}
+#endif
+
+	return 1;
+}
+
+/* Check for invalid option argumengts. */
+static int check_actions_arguments(struct lib_context *lc)
+{
+	if (valid_format(lc, OPT_STR_FORMAT(lc)))
+		return 1;
+
+	LOG_ERR(lc, 0, "Invalid format for -f at (see -l)");
+}
+
+/* Parse and handle the command line arguments */
+int handle_args(struct lib_context *lc, int argc, char ***argv)
+{
+	int o, ret = 0;
+#ifdef HAVE_GETOPTLONG
+	int opt_idx;
+#endif
+
+	if (argc < 2)
+		LOG_ERR(lc, 0, "No arguments/options given (-h for help)\n");
+
+#ifdef HAVE_GETOPTLONG
+	/* Walk the options (and option arguments) */
+	while ((o = getopt_long(argc, *argv, short_opts,
+				long_opts, &opt_idx)) != -1) {
+#else
+	while ((o = getopt(argc, *argv, short_opts)) != -1) {
+#endif
+		/* Help already displayed -> exit ok. */
+		if ((ret = set_action(lc, o)) && (HELP & action))
+			return 1;
+
+		if (!ret || o == ':' || o == '?')
+			return 0;
+	}
+
+	/* Force deactivation of stacked partition devices. */
+	/* FIXME: remove partiton code in favour of kpartx ? */
+	if (DEACTIVATE & action)
+		action &= ~NOPARTITIONS;
+
+	if ((ret = check_actions(lc, *argv)) && OPT_FORMAT(lc))
+		ret = check_actions_arguments(lc);
+
+	*argv += optind;
+
+	return ret;
+}
+
+static int version(struct lib_context *lc, int arg)
+{
+	char v[80];
+
+	dm_version(lc, v, sizeof(v));
+	log_print(lc, "%s version:\t\t%s\n"
+		      "%s library version:\t%s %s\n"
+		      "device-mapper version:\t%s",
+		      lc->cmd, DMRAID_VERSION,
+		      lc->cmd, libdmraid_version(lc), libdmraid_date(lc), v);
+
+	return 1;
+}
+
+/*********************************************************************
+ * Perform pre/post functions for requested actions.
+ */
+/* Post Activate/Deactivate RAID set. */
+#ifndef DMRAID_MINI
+/* Pre and post display_set() functions. */
+static int _display_sets_arg(int arg)
+{
+	return (action & ACTIVE) ?
+	       D_ACTIVE : ((action & INACTIVE) ? D_INACTIVE : D_ALL);
+}
+
+static int _display_set(struct lib_context *lc, void *rs, int type)
+{
+	display_set(lc, rs, type, 0);
+
+	return 1;
+}
+
+static int _display_sets(struct lib_context *lc, int type)
+{
+	process_sets(lc, _display_set, type, SETS);
+
+	return 1;
+}
+
+static int _display_devices(struct lib_context *lc, int type)
+{
+	display_devices(lc, type);
+
+	return 1;
+}
+
+static int _erase(struct lib_context *lc, int arg)
+{
+	return erase_metadata(lc);
+}
+#endif
+
+/* Retrieve and build metadata. */
+static int get_metadata(struct lib_context *lc, struct prepost *p, char **argv)
+{
+	if (!(M_DEVICE & p->metadata))
+		return 1;
+
+	if (!discover_devices(lc, OPT_DEVICES(lc) ? argv : NULL))
+		LOG_ERR(lc, 0, "failed to discover devices");
+
+	if(!count_devices(lc, DEVICE)) {
+		log_print(lc, "no block devices found");
+		return 1;
+	}
+
+	if (!(M_RAID & p->metadata))
+		return 1;
+
+#ifndef	DMRAID_MINI
+	/* Discover RAID disks and keep RAID metadata (eg, hpt45x) */
+	discover_raid_devices(lc, 
+# ifdef	DMRAID_NATIVE_LOG
+		    ((NATIVE_LOG|RAID_DEVICES) & action) ? argv : NULL);
+# else
+		    (RAID_DEVICES & action) ? argv : NULL);
+# endif
+#else
+	discover_raid_devices(lc, NULL);
+#endif
+	if (!count_devices(lc, RAID)) {
+		format_error(lc, "disks", argv);
+		return 1;
+	}
+
+	if (M_SET & p->metadata) {
+		/* Group RAID sets. */
+		build_sets(lc, argv);
+		if (!count_devices(lc, SET)) {
+			format_error(lc, "sets", argv);
+			return 0;
+		}
+	}
+
+	return 1;
+}
+
+/*
+ * Function abstraction which takes pre- and post-function calls
+ * to prepare an argument in pre() to be used by post().
+ *
+ * perform() is the call handler for all functions which need metadata
+ * as displaying, erasing and activation/deactivation of RAID sets.
+ *
+ * The necessary metadata describing disks, RAID devices and RAID sets
+ * gets automatically generated by this function.
+ *
+ * A lock gets taken out in case of metadata accesses in order to
+ * prevent multiple tool runs from occurring in parallel.
+ * For now I just lock globally, which will change when I get to monitoring
+ * of RAID sets, where finer grained locks on RAID sets need to be taken out.
+ */
+
+/*
+ * Definition of pre- and post functions to perform.
+ */
+struct prepost prepost[] = {
+	/* (De)activate RAID set. */
+	{ ACTIVATE|DEACTIVATE,
+	  M_DEVICE|M_RAID|M_SET,
+	  ROOT,
+	  LOCK,
+	  NULL,
+	  0,
+	  activate_or_deactivate_sets,
+	},
+
+#ifndef DMRAID_MINI
+	/* Display block devices. */
+	{ BLOCK_DEVICES,
+	  M_DEVICE,
+	  ROOT,
+	  NO_LOCK,
+	  NULL,
+	  DEVICE,
+	  _display_devices,
+	},
+
+	/* Erase metadata. */
+	{ ERASE,
+	  M_DEVICE|M_RAID, 
+	  ROOT,
+	  LOCK,
+	  NULL,
+	  0,
+	  _erase,
+	},
+
+	/* List metadata format handlers. */
+	{ LIST_FORMATS,
+	  M_NONE, 
+	  ANY_ID,
+	  NO_LOCK,
+	  NULL,
+	  0,
+	  list_formats,
+	},
+
+#  ifdef DMRAID_NATIVE_LOG
+	/* Native metadata log. */
+	{ NATIVE_LOG,
+	  M_DEVICE|M_RAID, 
+	  ROOT,
+	  LOCK,
+	  NULL,
+	  NATIVE,
+	  _display_devices,
+	},
+#  endif
+
+	/* Display RAID devices. */
+	{ RAID_DEVICES,
+	  M_DEVICE|M_RAID,
+	  ROOT,
+	  LOCK,
+	  NULL,
+	  RAID,
+	  _display_devices,
+	},
+
+	/* Display RAID sets. */
+	{ RAID_SETS,
+	  M_DEVICE|M_RAID|M_SET,
+	  ROOT,
+	  LOCK,
+	  _display_sets_arg,
+	  0,
+	  _display_sets,
+	},
+#endif
+
+	/* Display version. */
+	{ VERSION,
+	  M_NONE,
+	  ANY_ID,
+	  NO_LOCK,
+	  NULL,
+	  0,
+	  version,
+	},
+};
+
+static int _perform(struct lib_context *lc, struct prepost *p,
+		    char **argv)
+{
+	int ret = 0;
+
+	if (ROOT == p->id && geteuid())
+		LOG_ERR(lc, 0, "you must be root");
+	
+	/* Lock against parallel runs. Resource NULL for now. */
+	if (LOCK == p->lock && !lock_resource(lc, NULL))
+		LOG_ERR(lc, 0, "lock failure");
+	
+	if (get_metadata(lc, p, argv))
+		ret = p->post(lc, p->pre ? p->pre(p->arg) : p->arg);
+	
+	if (LOCK == p->lock)
+		unlock_resource(lc, NULL);
+	
+	return ret;
+}
+
+int perform(struct lib_context *lc, char **argv)
+{
+	struct prepost *p;
+
+	/* Special case, because help can be asked for at any time. */
+	if (HELP & action)
+		return 1;
+
+	/* Find appropriate action. */
+	for (p = prepost; p < ARRAY_END(prepost); p++) {
+		if (p->action & action)
+			return _perform(lc, p, argv);
+	}
+
+	return 0;
+}
/cvs/dm/dmraid/tools/commands.h,v  -->  standard output
revision 1.1
--- dmraid/tools/commands.h
+++ -	2008-02-22 16:57:45.431432000 +0000
@@ -0,0 +1,112 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef	_COMMANDS_H
+#define	_COMMANDS_H
+
+#include <dmraid/lib_context.h>
+
+#define ARRAY_SIZE(a)	(sizeof(a) / sizeof(*a))
+#define ARRAY_END(a)	(a + ARRAY_SIZE(a))
+
+/* Options actions dmraid performs. */
+enum action {
+	UNDEF		= 0x0,
+	ACTIVATE	= 0x1,
+	DEACTIVATE	= 0x2,
+	FORMAT		= 0x4,
+#ifndef	DMRAID_MINI
+	BLOCK_DEVICES	= 0x8,
+	COLUMN		= 0x10,
+	DBG		= 0x20,
+	DUMP		= 0x40,
+	ERASE		= 0x80,
+	GROUP		= 0x100,
+#endif
+	HELP		= 0x200,
+#ifndef	DMRAID_MINI
+	LIST_FORMATS	= 0x400,
+#  ifdef	DMRAID_NATIVE_LOG
+	NATIVE_LOG	= 0x800,
+#  endif
+#endif
+	NOPARTITIONS	= 0x1000,
+#ifndef	DMRAID_MINI
+	RAID_DEVICES	= 0x2000,
+	RAID_SETS	= 0x4000,
+	TEST		= 0x8000,
+	VERBOSE		= 0x10000,
+	ACTIVE		= 0x20000,
+	INACTIVE	= 0x40000,
+	SEPARATOR	= 0x80000,
+#endif
+	VERSION		= 0x100000,
+	IGNORELOCKING	= 0x200000,
+};
+
+#define	ALL_FLAGS	((enum action) -1)
+
+/* Arguments allowed ? */
+enum args {
+	NO_ARGS,
+	ARGS,
+};
+
+/*
+ * Action flag definitions for set_action().
+ *
+ * 'Early' options can be handled directly in set_action() by calling
+ * the functions registered here (f_set member) handing in arg.
+ */
+struct actions {
+	int option;		/* Option character/value. */
+	enum action action;	/* Action flag for this option or UNDEF. */
+	enum action needed;	/* Mandatory options or UNDEF if alone */
+	enum action allowed;	/* Allowed flags (ie, other options allowed) */
+
+	enum args args;		/* Arguments allowed ? */
+
+	/* Function to call on hit or NULL */
+	int (*f_set)(struct lib_context *lc, int arg);
+	int arg;		/* Argument for above function call */
+};
+
+/* Define which metadata is needed before we can call post functions. */
+enum metadata_need {
+	M_NONE	 = 0x00,
+	M_DEVICE = 0x01,
+	M_RAID	 = 0x02,
+	M_SET	 = 0x04,
+};
+
+enum id {
+	ROOT,
+	ANY_ID,
+};
+
+enum lock {
+	LOCK,
+	NO_LOCK,
+};
+
+/* 
+ * Pre and Post functions to perform for an option.
+ */
+struct prepost {
+	enum action action;
+	enum metadata_need metadata;
+	enum id id;
+	enum lock lock;
+	int (*pre)(int arg);
+	int arg;
+	int (*post)(struct lib_context *lc, int arg);
+};
+
+int handle_args(struct lib_context *lc, int argc, char ***argv);
+int perform(struct lib_context *lc, char **argv);
+
+#endif
/cvs/dm/dmraid/tools/dmraid.c,v  -->  standard output
revision 1.1
--- dmraid/tools/dmraid.c
+++ -	2008-02-22 16:57:45.524844000 +0000
@@ -0,0 +1,44 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * See commands.[ch] for the perform() function call abstraction below.
+ */
+
+#include <dmraid/dmraid.h>
+#include "commands.h"
+#include "toollib.h"
+#include "version.h"
+
+int main(int argc, char **argv)
+{
+	int ret = 0;
+	struct lib_context *lc;
+
+	/* Initialize library (creates a context to use it). */
+	if ((lc = libdmraid_init(argc, argv))) {
+		/*
+		 * Parse command line arguments and run 'early'
+		 * functions for options which set library context
+		 * variables (eg, --debug).
+		 *
+		 * Initialize locking afterwards, so that the
+		 * '--ignorelocking' option can be recognized.
+		 *
+		 * If both are ok -> perform the required action.
+		 */
+		ret = handle_args(lc, argc, &argv) &&
+		      init_locking(lc) &&
+		      perform(lc, argv);
+
+		/* Cleanup the library context. */
+		libdmraid_exit(lc);
+	}
+
+	/* Set standard exit code. */
+	exit (ret ? EXIT_SUCCESS : EXIT_FAILURE);
+}
/cvs/dm/dmraid/tools/toollib.c,v  -->  standard output
revision 1.1
--- dmraid/tools/toollib.c
+++ -	2008-02-22 16:57:45.615319000 +0000
@@ -0,0 +1,150 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+/*
+ * Tool library
+ */
+
+#include <dmraid/dmraid.h>
+#include "../lib/log/log.h"
+#include <ctype.h>
+
+#ifndef __KLIBC__
+# include <getopt.h>
+#endif
+
+#include <stdlib.h>
+#include <string.h>
+#include "commands.h"
+#include "toollib.h"
+
+/* [De]activate a RAID set. */
+static int _change_set(struct lib_context *lc, void *rs, int arg)
+{
+	if (change_set(lc,
+		       (ACTIVATE & action) ? A_ACTIVATE : A_DEACTIVATE,
+		       rs)) {
+		log_info(lc, "%sctivating %s RAID set \"%s\"",
+			 action & ACTIVATE ? "A": "Dea",
+			 get_set_type(lc, rs), get_set_name(lc, rs));
+		return 1;
+	}
+
+	return 0;
+}
+
+/* [De]activate RAID sets. */
+/* FIXME: remove partition code in favour of kpartx ? */
+static void process_partitions(struct lib_context *lc)
+{
+	discover_partitions(lc);
+	process_sets(lc, _change_set, 0, PARTITIONS);
+}
+
+int activate_or_deactivate_sets(struct lib_context *lc, int arg)
+{
+	/* Discover partitions to deactivate RAID sets for and work on them. */
+	if (DEACTIVATE & action)
+		process_partitions(lc);
+
+	process_sets(lc, _change_set, arg, SETS);
+
+	/* Discover partitions to activate RAID sets for and work on them. */
+	if ((ACTIVATE & action) && !(NOPARTITIONS & action))
+		process_partitions(lc);
+
+	return 1;
+}
+
+/* Build all sets or the ones given. */
+void build_sets(struct lib_context *lc, char **sets)
+{
+	int o = 0;
+
+	do {
+		if (!group_set(lc, sets[o]))
+			log_err(lc, "building set");
+
+		if (!sets[o])
+			break;
+	} while (sets[++o]);
+}
+
+/* Convert a character string to lower case. */
+void str_tolower(char *s)
+{
+	for (; *s; s++)
+		*s = tolower(*s);
+}
+
+/*
+ * Check if selected or all formats shall be used to read the metadata.
+*/
+/* Collapse a delimiter into one. */
+char *collapse_delimiter(struct lib_context *lc, char *str,
+			 size_t size, const char delim)
+{
+	size_t len;
+	char *p = str;
+
+	while ((p = strchr(p, delim))) {
+		if (p == str || p[1] == delim || !p[1])
+			memmove(p, p + 1, (len = str + size - p)), p[len] = 0;
+		else
+			p++;
+	}
+
+	return str;
+}
+
+int valid_format(struct lib_context *lc, const char *fmt)
+{
+	int ret = 1;
+	char *p, *p_sav, *sep;
+	const char delim = *OPT_STR_SEPARATOR(lc);
+
+ 	if (!(p_sav = dbg_strdup((char*) fmt)))
+		return log_alloc_err(lc, __func__);
+
+	sep = p_sav;
+	do {
+		sep = remove_delimiter((p = sep), delim);
+		log_notice(lc, "checking format identifier %s", p);
+
+		if (!(ret = check_valid_format(lc, p)))
+			break;
+
+		add_delimiter(&sep, delim);
+	} while (sep);
+
+	dbg_free(p_sav);
+
+	return ret;
+}
+
+void format_error(struct lib_context *lc, const char *error, char **argv)
+{
+	log_print_nnl(lc, "No RAID %s", error);
+
+	if (OPT_FORMAT(lc))
+		log_print_nnl(lc, " with format: \"%s\"",
+			      OPT_STR_FORMAT(lc));
+
+
+	if (argv && *argv) {
+		log_print_nnl(lc, " and with names: \"");
+		while (*argv) {
+			log_print_nnl(lc, "%s", *argv++);
+			if (*argv)
+				log_print_nnl(lc, "%s", OPT_STR_SEPARATOR(lc));
+			else
+				log_print_nnl(lc, "\"");
+		}
+	}
+
+	log_print(lc, "");
+}
/cvs/dm/dmraid/tools/toollib.h,v  -->  standard output
revision 1.1
--- dmraid/tools/toollib.h
+++ -	2008-02-22 16:57:45.715427000 +0000
@@ -0,0 +1,21 @@
+/*
+ * Copyright (C) 2004,2005  Heinz Mauelshagen, Red Hat GmbH.
+ *                          All rights reserved.
+ *
+ * See file LICENSE at the top of this source tree for license information.
+ */
+
+#ifndef _TOOLLIB_H_
+#define _TOOLLIB_H_
+
+extern enum action action;
+
+int  activate_or_deactivate_sets(struct lib_context *lc, int arg);
+void build_sets(struct lib_context *lc, char **sets);
+void format_error(struct lib_context *lc, const char *error, char **argv);
+void str_tolower(char *s);
+char *collapse_delimiter(struct lib_context *lc, char *str,
+			 size_t size, const char delim);
+int  valid_format(struct lib_context *lc, const char *fmt);
+
+#endif
/cvs/dm/dmraid/tools/version.h.in,v  -->  standard output
revision 1.1
--- dmraid/tools/version.h.in
+++ -	2008-02-22 16:57:45.816478000 +0000
@@ -0,0 +1,3 @@
+#ifndef	DMRAID_VERSION
+#define	DMRAID_VERSION	"@DMRAID_VERSION@ @FLAVOUR@"
+#endif




More information about the dm-devel mailing list