1*5bcb7424SSimon J. Gerraty# $Id: meta.autodep.mk,v 1.35 2014/05/09 00:05:46 sjg Exp $ 23cbdda60SSimon J. Gerraty 33cbdda60SSimon J. Gerraty# 43cbdda60SSimon J. Gerraty# @(#) Copyright (c) 2010, Simon J. Gerraty 53cbdda60SSimon J. Gerraty# 63cbdda60SSimon J. Gerraty# This file is provided in the hope that it will 73cbdda60SSimon J. Gerraty# be of use. There is absolutely NO WARRANTY. 83cbdda60SSimon J. Gerraty# Permission to copy, redistribute or otherwise 93cbdda60SSimon J. Gerraty# use this file is hereby granted provided that 103cbdda60SSimon J. Gerraty# the above copyright notice and this notice are 113cbdda60SSimon J. Gerraty# left intact. 123cbdda60SSimon J. Gerraty# 133cbdda60SSimon J. Gerraty# Please send copies of changes and bug-fixes to: 143cbdda60SSimon J. Gerraty# sjg@crufty.net 153cbdda60SSimon J. Gerraty# 163cbdda60SSimon J. Gerraty 173cbdda60SSimon J. Gerraty_this ?= ${.PARSEFILE} 183cbdda60SSimon J. Gerraty.if !target(__${_this}__) 193cbdda60SSimon J. Gerraty__${_this}__: .NOTMAIN 203cbdda60SSimon J. Gerraty 213cbdda60SSimon J. Gerraty.-include "local.autodep.mk" 223cbdda60SSimon J. Gerraty 233cbdda60SSimon J. Gerraty.if defined(SRCS) 243cbdda60SSimon J. Gerraty# it would be nice to be able to query .SUFFIXES 253cbdda60SSimon J. GerratyOBJ_EXTENSIONS+= .o .po .lo .So 263cbdda60SSimon J. Gerraty 273cbdda60SSimon J. Gerraty# explicit dependencies help short-circuit .SUFFIX searches 283cbdda60SSimon J. GerratySRCS_DEP_FILTER+= N*.[hly] 293cbdda60SSimon J. Gerraty.for s in ${SRCS:${SRCS_DEP_FILTER:O:u:ts:}} 303cbdda60SSimon J. Gerraty.for e in ${OBJ_EXTENSIONS:O:u} 313cbdda60SSimon J. Gerraty.if !target(${s:T:R}$e) 323cbdda60SSimon J. Gerraty${s:T:R}$e: $s 333cbdda60SSimon J. Gerraty.endif 343cbdda60SSimon J. Gerraty.endfor 353cbdda60SSimon J. Gerraty.endfor 363cbdda60SSimon J. Gerraty.endif 373cbdda60SSimon J. Gerraty 383cbdda60SSimon J. Gerraty.if make(gendirdeps) 393cbdda60SSimon J. Gerraty# you are supposed to know what you are doing! 403cbdda60SSimon J. GerratyUPDATE_DEPENDFILE = yes 413cbdda60SSimon J. Gerraty.elif !empty(.TARGETS) && !make(all) 423cbdda60SSimon J. Gerraty# do not update the *depend* files 433cbdda60SSimon J. Gerraty# unless we are building the entire directory or the default target. 443cbdda60SSimon J. Gerraty# NO means don't update .depend - or Makefile.depend* 453cbdda60SSimon J. Gerraty# no means update .depend but not Makefile.depend* 463cbdda60SSimon J. GerratyUPDATE_DEPENDFILE = NO 473cbdda60SSimon J. Gerraty.elif ${.MAKEFLAGS:M-k} != "" 483cbdda60SSimon J. Gerraty# it is a bad idea to update anything 493cbdda60SSimon J. GerratyUPDATE_DEPENDFILE = NO 503cbdda60SSimon J. Gerraty.endif 513cbdda60SSimon J. Gerraty 523cbdda60SSimon J. Gerraty_CURDIR ?= ${.CURDIR} 533cbdda60SSimon J. Gerraty_DEPENDFILE := ${_CURDIR}/${.MAKE.DEPENDFILE:T} 543cbdda60SSimon J. Gerraty 553cbdda60SSimon J. Gerraty.if ${.MAKE.LEVEL} == 0 563cbdda60SSimon J. Gerraty.if ${BUILD_AT_LEVEL0:Uyes:tl} == "no" 573cbdda60SSimon J. GerratyUPDATE_DEPENDFILE = NO 583cbdda60SSimon J. Gerraty.endif 593cbdda60SSimon J. Gerraty.endif 603cbdda60SSimon J. Gerraty.if !exists(${_DEPENDFILE}) 613cbdda60SSimon J. Gerraty_bootstrap_dirdeps = yes 623cbdda60SSimon J. Gerraty.endif 633cbdda60SSimon J. Gerraty_bootstrap_dirdeps ?= no 643cbdda60SSimon J. GerratyUPDATE_DEPENDFILE ?= yes 653cbdda60SSimon J. Gerraty 663cbdda60SSimon J. Gerraty.if ${DEBUG_AUTODEP:Uno:@m@${RELDIR:M$m}@} != "" 673cbdda60SSimon J. Gerraty.info ${_DEPENDFILE:S,${SRCTOP}/,,} update=${UPDATE_DEPENDFILE} 683cbdda60SSimon J. Gerraty.endif 693cbdda60SSimon J. Gerraty 703cbdda60SSimon J. Gerraty.if !empty(XMAKE_META_FILE) 713cbdda60SSimon J. Gerraty.if exists(${.OBJDIR}/${XMAKE_META_FILE}) 723cbdda60SSimon J. Gerraty# we cannot get accurate dependencies from an update build 733cbdda60SSimon J. GerratyUPDATE_DEPENDFILE = NO 743cbdda60SSimon J. Gerraty.else 753cbdda60SSimon J. GerratyMETA_XTRAS += ${XMAKE_META_FILE} 763cbdda60SSimon J. Gerraty.endif 773cbdda60SSimon J. Gerraty.endif 783cbdda60SSimon J. Gerraty 793cbdda60SSimon J. Gerraty.if ${_bootstrap_dirdeps} == "yes" || exists(${_DEPENDFILE}) 803cbdda60SSimon J. Gerraty# if it isn't supposed to be touched by us the Makefile should have 813cbdda60SSimon J. Gerraty# UPDATE_DEPENDFILE = no 823cbdda60SSimon J. GerratyWANT_UPDATE_DEPENDFILE ?= yes 833cbdda60SSimon J. Gerraty.endif 843cbdda60SSimon J. Gerraty 853cbdda60SSimon J. Gerraty.if ${WANT_UPDATE_DEPENDFILE:Uno:tl} != "no" 863cbdda60SSimon J. Gerraty.if ${.MAKE.MODE:Mmeta*} == "" || ${.MAKE.MODE:M*read*} != "" 873cbdda60SSimon J. GerratyUPDATE_DEPENDFILE = no 883cbdda60SSimon J. Gerraty.endif 893cbdda60SSimon J. Gerraty 903cbdda60SSimon J. Gerraty.if ${DEBUG_AUTODEP:Uno:@m@${RELDIR:M$m}@} != "" 913cbdda60SSimon J. Gerraty.info ${_DEPENDFILE:S,${SRCTOP}/,,} update=${UPDATE_DEPENDFILE} 923cbdda60SSimon J. Gerraty.endif 933cbdda60SSimon J. Gerraty 943cbdda60SSimon J. Gerraty.if ${UPDATE_DEPENDFILE:tl} == "yes" 953cbdda60SSimon J. Gerraty# sometimes we want .meta files generated to aid debugging/error detection 963cbdda60SSimon J. Gerraty# but do not want to consider them for dependencies 973cbdda60SSimon J. Gerraty# for example the result of running configure 983cbdda60SSimon J. Gerraty# just make sure this is not empty 993cbdda60SSimon J. GerratyMETA_FILE_FILTER ?= N.meta 1003cbdda60SSimon J. Gerraty 1013cbdda60SSimon J. Gerraty.if !empty(DPADD) 1023cbdda60SSimon J. Gerraty# if we have any non-libs in DPADD, 1033cbdda60SSimon J. Gerraty# they probably need to be paid attention to 1043cbdda60SSimon J. Gerraty.if !empty(DPLIBS) 1053cbdda60SSimon J. GerratyFORCE_DPADD = ${DPADD:${DPLIBS:${M_ListToSkip}}:${DPADD_LAST:${M_ListToSkip}}} 1063cbdda60SSimon J. Gerraty.else 1073cbdda60SSimon J. Gerraty_nonlibs := ${DPADD:T:Nlib*:N*include} 1083cbdda60SSimon J. Gerraty.if !empty(_nonlibs) 1093cbdda60SSimon J. GerratyFORCE_DPADD += ${_nonlibs:@x@${DPADD:M*/$x}@} 1103cbdda60SSimon J. Gerraty.endif 1113cbdda60SSimon J. Gerraty.endif 1123cbdda60SSimon J. Gerraty.endif 1133cbdda60SSimon J. Gerraty 1143cbdda60SSimon J. Gerraty.if !make(gendirdeps) 1153cbdda60SSimon J. Gerraty.END: gendirdeps 1163cbdda60SSimon J. Gerraty.endif 1173cbdda60SSimon J. Gerraty 1183cbdda60SSimon J. Gerraty# if we don't have OBJS, then .depend isn't useful 1193cbdda60SSimon J. Gerraty.if !target(.depend) && (!empty(OBJS) || ${.ALLTARGETS:M*.o} != "") 1203cbdda60SSimon J. Gerraty# some makefiles and/or targets contain 1213cbdda60SSimon J. Gerraty# circular dependencies if you dig too deep 1223cbdda60SSimon J. Gerraty# (as meta mode is apt to do) 123*5bcb7424SSimon J. Gerraty# so we provide a means of suppressing them. 1243cbdda60SSimon J. Gerraty# the input to the loop below is target: dependency 1253cbdda60SSimon J. Gerraty# with just one dependency per line. 1263cbdda60SSimon J. Gerraty# Also some targets are not really local, or use random names. 1273cbdda60SSimon J. Gerraty# Use local.autodep.mk to provide local additions! 1283cbdda60SSimon J. GerratySUPPRESS_DEPEND += \ 1293cbdda60SSimon J. Gerraty ${SB:S,/,_,g}* \ 1303cbdda60SSimon J. Gerraty *:y.tab.c \ 1313cbdda60SSimon J. Gerraty *.c:*.c \ 1323cbdda60SSimon J. Gerraty *.h:*.h 1333cbdda60SSimon J. Gerraty 1343cbdda60SSimon J. Gerraty.NOPATH: .depend 1353cbdda60SSimon J. Gerraty# we use ${.MAKE.META.CREATED} to trigger an update but 1363cbdda60SSimon J. Gerraty# we process using ${.MAKE.META.FILES} 1373cbdda60SSimon J. Gerraty# the double $$ defers initial evaluation 1383cbdda60SSimon J. Gerraty# if necessary, we fake .po dependencies, just so the result 1393cbdda60SSimon J. Gerraty# in Makefile.depend* is stable 140*5bcb7424SSimon J. Gerraty# The current objdir may be referred to in various ways 1413cbdda60SSimon J. GerratyOBJDIR_REFS += ${.OBJDIR} ${.OBJDIR:tA} ${_OBJDIR} ${RELOBJTOP}/${RELDIR} 1423cbdda60SSimon J. Gerraty_depend = .depend 1433cbdda60SSimon J. Gerraty# it would be nice to be able to get .SUFFIXES as ${.SUFFIXES} 1443cbdda60SSimon J. Gerraty# we actually only care about the .SUFFIXES of files that might be 1453cbdda60SSimon J. Gerraty# generated by tools like yacc. 1463cbdda60SSimon J. GerratyDEPEND_SUFFIXES += .c .h .cpp .hpp .cxx .hxx .cc .hh 1473cbdda60SSimon J. Gerraty.depend: .NOMETA $${.MAKE.META.CREATED} ${_this} 1483cbdda60SSimon J. Gerraty @echo "Updating $@: ${.OODATE:T:[1..8]}" 1493cbdda60SSimon J. Gerraty @egrep -i '^R .*\.(${DEPEND_SUFFIXES:tl:O:u:S,^.,,:ts|})$$' /dev/null ${.MAKE.META.FILES:T:O:u:${META_FILE_FILTER:ts:}:M*o.meta} | \ 1503cbdda60SSimon J. Gerraty sed -e 's, \./, ,${OBJDIR_REFS:O:u:@d@;s, $d/, ,@};/\//d' \ 1513cbdda60SSimon J. Gerraty -e 's,^\([^/][^/]*\).meta...[0-9]* ,\1: ,' | \ 1523cbdda60SSimon J. Gerraty sort -u | \ 1533cbdda60SSimon J. Gerraty while read t d; do \ 1543cbdda60SSimon J. Gerraty case "$$d:" in $$t) continue;; esac; \ 1553cbdda60SSimon J. Gerraty case "$$t$$d" in ${SUPPRESS_DEPEND:U.:O:u:ts|}) continue;; esac; \ 1563cbdda60SSimon J. Gerraty echo $$t $$d; \ 1573cbdda60SSimon J. Gerraty done > $@.${.MAKE.PID} 1583cbdda60SSimon J. Gerraty @case "${.MAKE.META.FILES:T:M*.po.*}" in \ 1593cbdda60SSimon J. Gerraty *.po.*) mv $@.${.MAKE.PID} $@;; \ 1603cbdda60SSimon J. Gerraty *) { cat $@.${.MAKE.PID}; \ 1613cbdda60SSimon J. Gerraty sed 's,\.So:,.o:,;s,\.o:,.po:,' $@.${.MAKE.PID}; } | sort -u > $@; \ 1623cbdda60SSimon J. Gerraty rm -f $@.${.MAKE.PID};; \ 1633cbdda60SSimon J. Gerraty esac 1643cbdda60SSimon J. Gerraty.else 1653cbdda60SSimon J. Gerraty# make sure this exists 1663cbdda60SSimon J. Gerraty.depend: 1673cbdda60SSimon J. Gerraty# do _not_ assume that .depend is in any fit state for us to use 1683cbdda60SSimon J. GerratyCAT_DEPEND = /dev/null 1693cbdda60SSimon J. Gerraty.if ${.MAKE.LEVEL} > 0 1703cbdda60SSimon J. Gerraty.export CAT_DEPEND 1713cbdda60SSimon J. Gerraty.endif 1723cbdda60SSimon J. Gerraty_depend = 1733cbdda60SSimon J. Gerraty.endif 1743cbdda60SSimon J. Gerraty 1753cbdda60SSimon J. Gerraty.if ${DEBUG_AUTODEP:Uno:@m@${RELDIR:M$m}@} != "" 1763cbdda60SSimon J. Gerraty.info ${_DEPENDFILE:S,${SRCTOP}/,,} _depend=${_depend} 1773cbdda60SSimon J. Gerraty.endif 1783cbdda60SSimon J. Gerraty 179*5bcb7424SSimon J. Gerraty.if ${UPDATE_DEPENDFILE} == "yes" 1803cbdda60SSimon J. Gerratygendirdeps: ${_DEPENDFILE} 181*5bcb7424SSimon J. Gerraty.endif 1823cbdda60SSimon J. Gerraty 1833cbdda60SSimon J. Gerraty.if !target(${_DEPENDFILE}) 1843cbdda60SSimon J. Gerraty.if ${_bootstrap_dirdeps} == "yes" 1853cbdda60SSimon J. Gerraty# We are boot-strapping a new directory 1863cbdda60SSimon J. Gerraty# Use DPADD to seed DIRDEPS 1873cbdda60SSimon J. Gerraty.if !empty(DPADD) 1883cbdda60SSimon J. Gerraty# anything which matches ${_OBJROOT}* but not ${_OBJTOP}* 1893cbdda60SSimon J. Gerraty# needs to be qualified in DIRDEPS 1903cbdda60SSimon J. Gerraty# The pseudo machine "host" is used for HOST_TARGET 1913cbdda60SSimon J. GerratyDIRDEPS = \ 1923cbdda60SSimon J. Gerraty ${DPADD:M${_OBJTOP}*:H:C,${_OBJTOP}[^/]*/,,:N.:O:u} \ 1933cbdda60SSimon J. Gerraty ${DPADD:M${_OBJROOT}*:N${_OBJTOP}*:H:S,${_OBJROOT},,:C,^([^/]+)/(.*),\2.\1,:S,${HOST_TARGET}$,host,:N.*:O:u} 1943cbdda60SSimon J. Gerraty 1953cbdda60SSimon J. Gerraty.endif 1963cbdda60SSimon J. Gerraty.endif 1973cbdda60SSimon J. Gerraty 1983cbdda60SSimon J. Gerraty_gendirdeps_mutex = 1993cbdda60SSimon J. Gerraty.if defined(NEED_GENDIRDEPS_MUTEX) 2003cbdda60SSimon J. Gerraty# If a src dir gets built with multiple object dirs, 2013cbdda60SSimon J. Gerraty# we need a mutex. Obviously, this is best avoided. 2023cbdda60SSimon J. Gerraty# Note if .MAKE.DEPENDFILE is common for all ${MACHINE} 2033cbdda60SSimon J. Gerraty# you either need to mutex, or ensure only one machine builds at a time! 2043cbdda60SSimon J. Gerraty# lockf is an example of a suitable tool 2053cbdda60SSimon J. GerratyLOCKF ?= /usr/bin/lockf 2063cbdda60SSimon J. Gerraty.if exists(${LOCKF}) 2073cbdda60SSimon J. GerratyGENDIRDEPS_MUTEXER ?= ${LOCKF} -k 2083cbdda60SSimon J. Gerraty.endif 2093cbdda60SSimon J. Gerraty.if empty(GENDIRDEPS_MUTEXER) 2103cbdda60SSimon J. Gerraty.error NEED_GENDIRDEPS_MUTEX defined, but GENDIRDEPS_MUTEXER not set 2113cbdda60SSimon J. Gerraty.else 2123cbdda60SSimon J. Gerraty_gendirdeps_mutex = ${GENDIRDEPS_MUTEXER} ${GENDIRDEPS_MUTEX:U${_CURDIR}/Makefile} 2133cbdda60SSimon J. Gerraty.endif 2143cbdda60SSimon J. Gerraty.endif 2153cbdda60SSimon J. Gerraty 2163cbdda60SSimon J. Gerraty# If we have META_XTRAS we most likely did not create them 2173cbdda60SSimon J. Gerraty# but we need to behave as if we did. 2183cbdda60SSimon J. Gerraty# Avoid adding glob patterns to .MAKE.META.CREATED though. 2193cbdda60SSimon J. Gerraty.MAKE.META.CREATED += ${META_XTRAS:N*\**:O:u} 2203cbdda60SSimon J. Gerraty 2213cbdda60SSimon J. Gerraty.if make(gendirdeps) 2223cbdda60SSimon J. GerratyMETA_FILES = *.meta 2233cbdda60SSimon J. Gerraty.elif ${OPTIMIZE_OBJECT_META_FILES:Uno:tl} == "no" 2243cbdda60SSimon J. GerratyMETA_FILES = ${.MAKE.META.FILES:T:N.depend*:O:u} 2253cbdda60SSimon J. Gerraty.else 2263cbdda60SSimon J. Gerraty# if we have 1000's of .o.meta, .So.meta etc we need only look at one set 2273cbdda60SSimon J. Gerraty# it is left as an exercise for the reader to work out what this does 2283cbdda60SSimon J. GerratyMETA_FILES = ${.MAKE.META.FILES:T:N.depend*:N*o.meta:O:u} \ 2293cbdda60SSimon J. Gerraty ${.MAKE.META.FILES:T:M*.${.MAKE.META.FILES:M*o.meta:R:E:O:u:[1]}.meta:O:u} 2303cbdda60SSimon J. Gerraty.endif 2313cbdda60SSimon J. Gerraty 2323cbdda60SSimon J. Gerraty.if ${DEBUG_AUTODEP:Uno:@m@${RELDIR:M$m}@} != "" 2333cbdda60SSimon J. Gerraty.info ${_DEPENDFILE:S,${SRCTOP}/,,}: ${_depend} ${.PARSEDIR}/gendirdeps.mk ${META2DEPS} xtras=${META_XTRAS} 2343cbdda60SSimon J. Gerraty.endif 2353cbdda60SSimon J. Gerraty 2363cbdda60SSimon J. Gerraty.if ${.MAKE.LEVEL} > 0 && !empty(GENDIRDEPS_FILTER) 2373cbdda60SSimon J. Gerraty.export GENDIRDEPS_FILTER 2383cbdda60SSimon J. Gerraty.endif 2393cbdda60SSimon J. Gerraty 2403cbdda60SSimon J. Gerraty# we might have .../ in MAKESYSPATH 2413cbdda60SSimon J. Gerraty_makesyspath:= ${_PARSEDIR} 2423cbdda60SSimon J. Gerraty${_DEPENDFILE}: ${_depend} ${.PARSEDIR}/gendirdeps.mk ${META2DEPS} $${.MAKE.META.CREATED} 2433cbdda60SSimon J. Gerraty @echo Checking $@: ${.OODATE:T:[1..8]} 2443cbdda60SSimon J. Gerraty @(cd . && \ 2453cbdda60SSimon J. Gerraty SKIP_GENDIRDEPS='${SKIP_GENDIRDEPS:O:u}' \ 2463cbdda60SSimon J. Gerraty DPADD='${FORCE_DPADD:O:u}' ${_gendirdeps_mutex} \ 2473cbdda60SSimon J. Gerraty MAKESYSPATH=${_makesyspath} \ 2483cbdda60SSimon J. Gerraty ${.MAKE} -f gendirdeps.mk RELDIR=${RELDIR} _DEPENDFILE=${_DEPENDFILE} \ 2493cbdda60SSimon J. Gerraty META_FILES='${META_XTRAS:T:O:u} ${META_FILES:T:O:u:${META_FILE_FILTER:ts:}}') 2503cbdda60SSimon J. Gerraty @test -s $@ && touch $@; : 2513cbdda60SSimon J. Gerraty.endif 2523cbdda60SSimon J. Gerraty 2533cbdda60SSimon J. Gerraty.endif 2543cbdda60SSimon J. Gerraty.endif 2553cbdda60SSimon J. Gerraty 2563cbdda60SSimon J. Gerraty.if ${_bootstrap_dirdeps} == "yes" 2573cbdda60SSimon J. Gerraty# make sure this is included at least once 2583cbdda60SSimon J. Gerraty.include <dirdeps.mk> 2593cbdda60SSimon J. Gerraty.else 2603cbdda60SSimon J. Gerraty${_DEPENDFILE}: .PRECIOUS 2613cbdda60SSimon J. Gerraty.endif 2623cbdda60SSimon J. Gerraty 2633cbdda60SSimon J. GerratyCLEANFILES += *.meta filemon.* *.db 264*5bcb7424SSimon J. Gerraty 265*5bcb7424SSimon J. Gerraty# these make it easy to gather some stats 266*5bcb7424SSimon J. Gerratynow_utc = ${%s:L:gmtime} 267*5bcb7424SSimon J. Gerratystart_utc := ${now_utc} 268*5bcb7424SSimon J. Gerraty 269*5bcb7424SSimon J. Gerratymeta_stats= meta=${empty(.MAKE.META.FILES):?0:${.MAKE.META.FILES:[#]}} \ 270*5bcb7424SSimon J. Gerraty created=${empty(.MAKE.META.CREATED):?0:${.MAKE.META.CREATED:[#]}} 271*5bcb7424SSimon J. Gerraty 272*5bcb7424SSimon J. Gerraty#.END: _reldir_finish 273*5bcb7424SSimon J. Gerraty.if target(gendirdeps) 274*5bcb7424SSimon J. Gerraty_reldir_finish: gendirdeps 275*5bcb7424SSimon J. Gerraty.endif 276*5bcb7424SSimon J. Gerraty_reldir_finish: .NOMETA 277*5bcb7424SSimon J. Gerraty @echo "${TIME_STAMP} Finished ${RELDIR}.${TARGET_SPEC} seconds=$$(( ${now_utc} - ${start_utc} )) ${meta_stats}" 278*5bcb7424SSimon J. Gerraty 279*5bcb7424SSimon J. Gerraty#.ERROR: _reldir_failed 280*5bcb7424SSimon J. Gerraty_reldir_failed: .NOMETA 281*5bcb7424SSimon J. Gerraty @echo "${TIME_STAMP} Failed ${RELDIR}.${TARGET_SPEC} seconds=$$(( ${now_utc} - ${start_utc} )) ${meta_stats}" 282*5bcb7424SSimon J. Gerraty 283*5bcb7424SSimon J. Gerraty.if defined(WITH_META_STATS) && ${.MAKE.LEVEL} > 0 284*5bcb7424SSimon J. Gerraty.END: _reldir_finish 285*5bcb7424SSimon J. Gerraty.ERROR: _reldir_failed 286*5bcb7424SSimon J. Gerraty.endif 287*5bcb7424SSimon J. Gerraty 2883cbdda60SSimon J. Gerraty.endif 289