# -*- tab-width: 4 -*- ;; Emacs
# vi: set filetype=sh tabstop=8 shiftwidth=8 noexpandtab :: Vi/ViM
############################################################ IDENT(1)
#
# $Title: dwatch(8) module for VOP_SYMLINK(9) [or similar] entry $
# $Copyright: 2014-2018 Devin Teske. All rights reserved. $
#
############################################################ DESCRIPTION
#
# Print symlink paths being created by VOP_SYMLINK(9) [or similar]
# NB: All paths are shown even if error prevents their creation.
#
############################################################ PROBE
: ${PROBE:=vfs:vop:$PROFILE:entry}
############################################################ ACTIONS
exec 9<<EOF
$PROBE /* probe ID $ID */
{${TRACE:+
printf("<$ID>");}
this->vp = (struct vnode *)arg0;
this->ncp = this->vp != NULL ?
this->vp->v_cache_dst.tqh_first : 0;
this->target = args[1] ? args[1]->a_target : "";
this->fi_name = args[1] ? (
args[1]->a_cnp != NULL ?
stringof(args[1]->a_cnp->cn_nameptr) : ""
) : "";
this->mount = this->vp != NULL ?
this->vp->v_mount : NULL; /* ptr to vfs we are in */
this->fi_fs = this->mount != NULL ?
stringof(this->mount->mnt_stat.f_fstypename) : "";
this->fi_mount = this->mount != NULL ?
stringof(this->mount->mnt_stat.f_mntonname) : "";
this->d_name = args[0]->v_cache_dd != NULL ?
stringof(args[0]->v_cache_dd->nc_name) : "";
$( awk -v MAX_DEPTH=$MAX_DEPTH '
{ sub(/^\\\t/, "\t") }
{ buf = buf "\t" $0 "\n" }
END {
sub(/\n$/, "", buf)
$0 = buf
sub(/^[[:space:]]*/, "")
for (DEPTH = 1; DEPTH <= MAX_DEPTH + 1; DEPTH++) {
gsub(/DEPTH/, DEPTH)
print
$0 = buf
}
}
' <<-EOFDEPTH
this->nameDEPTH = "";
EOFDEPTH
)
}
$PROBE /this->vp == 0 || this->fi_fs == 0 ||
this->fi_fs == "devfs" || this->fi_fs == "" ||
this->fi_name == ""/ /* probe ID $(( $ID + 1 )) */
{${TRACE:+
printf("<$(( $ID + 1 ))>");}
this->ncp = 0;
}
/*********************************************************/
$PROBE /this->ncp/ /* probe ID $(( $ID + 2 )) (depth 1) */
{${TRACE:+
printf("<$(( $ID + 2 ))>");}
this->dvp = this->ncp->nc_dvp != NULL ?
this->ncp->nc_dvp->v_cache_dst.tqh_first : 0;
this->name1 = this->dvp != 0 ? (
this->dvp->nc_name != 0 ? stringof(this->dvp->nc_name) : ""
) : "";
}
$PROBE /this->name1 == 0 || this->fi_fs == 0 ||
this->fi_fs == "devfs" || this->fi_fs == "" ||
this->name1 == "/" || this->name1 == ""/ /* probe ID $(( $ID + 3 )) */
{${TRACE:+
printf("<$(( $ID + 3 ))>");}
this->dvp = 0;
}
/*********************************************************/
/*
* BEGIN Pathname-depth iterators
*/
$( awk -v ID=$(( $ID + 4 )) -v MAX_DEPTH=$MAX_DEPTH '
{ buf = buf $0 "\n" }
END {
sub(/\n$/, "", buf)
for (DEPTH = 2; DEPTH <= MAX_DEPTH; DEPTH++) {
$0 = buf
gsub(/DEPTH/, DEPTH)
gsub(/IDNUM/, ID++)
print
}
}
' <<EOFDEPTH
$PROBE /this->dvp/ /* probe ID IDNUM (depth DEPTH) */
{${TRACE:+
printf("<IDNUM>");}
this->dvp = this->dvp->nc_dvp != NULL ?
this->dvp->nc_dvp->v_cache_dst.tqh_first : 0;
this->nameDEPTH = this->dvp != 0 ? (
this->dvp->nc_name != 0 ? stringof(this->dvp->nc_name) : ""
) : "";
}
EOFDEPTH
)
$PROBE /this->dvp/ /* probe ID $(( $ID + $MAX_DEPTH + 3 )) */
{${TRACE:+
printf("<$(( $ID + $MAX_DEPTH + 3 ))>");}
this->dvp = this->dvp->nc_dvp != NULL ?
this->dvp->nc_dvp->v_cache_dst.tqh_first : 0;
this->name$(( $MAX_DEPTH + 1 )) = this->dvp != 0 ? (
this->dvp->nc_dvp != NULL ? "..." : ""
) : "";
}
/*
* END Pathname-depth iterators
*/
/*********************************************************/
$PROBE /this->fi_mount != 0/ /* probe ID $(( $ID + $MAX_DEPTH + 4 )) */
{${TRACE:+
printf("<$(( $ID + $MAX_DEPTH + 4 ))>");
}
/*
* Join full path
* NB: Up-to but not including the parent directory (joined below)
*/
this->path = this->fi_mount;
this->path = strjoin(this->path, this->fi_mount != 0 ? (
this->fi_mount == "/" ? "" : "/"
) : "/");
$( awk -v MAX_DEPTH=$MAX_DEPTH '
{ sub(/^\\\t/, "\t") }
{ buf = buf "\t" $0 "\n" }
END {
sub(/\n$/, "", buf)
$0 = buf
sub(/^[[:space:]]*/, "")
for (N = MAX_DEPTH + 1; N > 0; N--) {
gsub(/N/, N)
print
$0 = buf
}
}
' <<-EOFDEPTH
this->path = strjoin(this->path,
\ strjoin(this->nameN, this->nameN != "" ? "/" : ""));
EOFDEPTH
)
/* Join the parent directory name */
this->path = strjoin(this->path, strjoin(this->name =
(this->d_name != 0 ? this->d_name : ""),
this->name != "" ? "/" : ""));
/* Join the entry name */
this->path = strjoin(this->path,
this->name = (this->fi_name != 0 ? this->fi_name : ""));
}
EOF
ACTIONS=$( cat <&9 )
ID=$(( $ID + $MAX_DEPTH + 5 ))
############################################################ EVENT ACTION
EVENT_TEST="this->fi_mount != 0"
############################################################ EVENT DETAILS
if [ ! "$CUSTOM_DETAILS" ]; then
exec 9<<EOF
/*
* Print full path and target
*/
printf("%s -> %s", this->path, this->target);
EOF
EVENT_DETAILS=$( cat <&9 )
fi
################################################################################
# END
################################################################################