sync code with last improvements from OpenBSD

This commit is contained in:
purplerain 2023-09-12 03:18:43 +00:00
parent 2a511f7966
commit 5455e77c6b
Signed by: purplerain
GPG Key ID: F42C07F07E2E35B7
13 changed files with 283 additions and 177 deletions

View File

@ -1,4 +1,4 @@
# $OpenBSD: Makefile,v 1.29 2023/09/02 19:28:46 dv Exp $ # $OpenBSD: Makefile,v 1.30 2023/09/11 19:03:53 mpi Exp $
BTRACE?= /usr/sbin/btrace BTRACE?= /usr/sbin/btrace
ALLOWDT!= sysctl -n kern.allowdt 2>/dev/null ALLOWDT!= sysctl -n kern.allowdt 2>/dev/null
@ -9,7 +9,7 @@ BT_LANG_SCRIPTS= arithm beginend beginend-argn boolean comments delete \
map mapclear mapempty mapsyntax mapzero map-unnamed \ map mapclear mapempty mapsyntax mapzero map-unnamed \
maxoperand min+max+sum multismts nsecs+var \ maxoperand min+max+sum multismts nsecs+var \
precedence print read-map-after-clear staticv-empty \ precedence print read-map-after-clear staticv-empty \
syntaxerror syntaxerror tuple tupleeval vareval
BT_ARG_LANG_SCRIPTS= staticv str BT_ARG_LANG_SCRIPTS= staticv str

View File

@ -0,0 +1,4 @@
BEGIN {
$t = (1, 42, "something");
printf("%d %s\n", $t.1, $t.2);
}

View File

@ -0,0 +1 @@
42 something

View File

@ -0,0 +1,8 @@
BEGIN {
$t = (5, comm, "str", probe);
@u = $t;
}
END {
print(@u)
}

View File

@ -0,0 +1 @@
(5, btrace, str, BEGIN)

View File

@ -0,0 +1,5 @@
BEGIN {
$tmp = 5;
@t = $tmp;
@m[42] = @t;
}

View File

@ -0,0 +1 @@
@m[42]: 5

View File

@ -1,4 +1,5 @@
.\" $OpenBSD: ruby-module.5,v 1.40 2023/05/31 15:27:45 jeremy Exp $ m
.\" $OpenBSD: ruby-module.5,v 1.41 2023/09/11 20:02:38 jeremy Exp $
.\" .\"
.\" Copyright (c) 2008 Marc Espie, Jeremy Evans .\" Copyright (c) 2008 Marc Espie, Jeremy Evans
.\" .\"
@ -24,7 +25,7 @@
.\" (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF .\" (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
.\" THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. .\" THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
.\" .\"
.Dd $Mdocdate: May 31 2023 $ .Dd $Mdocdate: September 11 2023 $
.Dt RUBY-MODULE 5 .Dt RUBY-MODULE 5
.Os .Os
.Sh NAME .Sh NAME
@ -38,48 +39,40 @@ in the
tree. tree.
.Pp .Pp
Sets Sets
.Ev MODRUBY_REV , .Ev GEM ,
.Ev RUBY , .Ev GEM_BASE_BIN ,
.Ev RAKE , .Ev GEM_BASE_LIB ,
.Ev RSPEC , .Ev GEM_BIN ,
.Ev MODRUBY_BIN_TESTRB , .Ev GEM_BIN_SUFFIX ,
.Ev MODRUBY_BIN_RSPEC , .Ev GEM_EXTENSIONS_DIR,
.Ev MODRUBY_WANTLIB , .Ev GEM_FLAGS ,
.Ev MODRUBY_RUN_DEPENDS , .Ev GEM_LIB ,
.Ev MODRUBY_LIB_DEPENDS , .Ev GEM_MAN_SUFFIX ,
.Ev MODRUBY_BUILD_DEPENDS ,
.Ev MODRUBY_RSPEC_DEPENDS ,
.Ev MODRUBY_RSPEC3_DEPENDS ,
.Ev MODRUBY_RUBY_ADJ ,
.Ev MODRUBY_ADJ_FILES , .Ev MODRUBY_ADJ_FILES ,
.Ev MODRUBY_LIBDIR , .Ev MODRUBY_ARCH ,
.Ev MODRUBY_DOCDIR , .Ev MODRUBY_BINREV ,
.Ev MODRUBY_BIN_RSPEC ,
.Ev MODRUBY_BUILD_DEPENDS ,
.Ev MODRUBY_BUILDDEP ,
.Ev MODRUBY_FLAVOR ,
.Ev MODRUBY_HANDLE_FLAVORS ,
.Ev MODRUBY_LIBREV ,
.Ev MODRUBY_LIB_DEPENDS ,
.Ev MODRUBY_PKG_PREFIX ,
.Ev MODRUBY_RELEXAMPLEDIR ,
.Ev MODRUBY_REV ,
.Ev MODRUBY_RUBY_ADJ ,
.Ev MODRUBY_RUN_DEPENDS ,
.Ev MODRUBY_RUNDEP ,
.Ev MODRUBY_SITEARCHDIR , .Ev MODRUBY_SITEARCHDIR ,
.Ev MODRUBY_SITEDIR , .Ev MODRUBY_SITEDIR ,
.Ev MODRUBY_EXAMPLEDIR ,
.Ev MODRUBY_ARCH ,
.Ev MODRUBY_LIBREV ,
.Ev MODRUBY_BINREV ,
.Ev MODRUBY_HANDLE_FLAVORS ,
.Ev MODRUBY_PKG_PREFIX ,
.Ev MODRUBY_FLAVOR ,
.Ev MODRUBY_TEST , .Ev MODRUBY_TEST ,
.Ev MODRUBY_TEST_TARGET ,
.Ev MODRUBY_TEST_DIR ,
.Ev MODRUBY_TEST_ENV , .Ev MODRUBY_TEST_ENV ,
.Ev MODRUBY_BUILD_TARGET , .Ev MODRUBY_TEST_TARGET ,
.Ev MODRUBY_INSTALL_TARGET , .Ev MODRUBY_WANTLIB ,
.Ev MODRUBY_EXTRACT_TARGET , .Ev RAKE ,
.Ev GEM ,
.Ev GEM_BIN ,
.Ev GEM_LIB ,
.Ev GEM_BASE_LIB ,
.Ev GEM_ABS_PATH ,
.Ev GEM_BASE_BIN ,
.Ev GEM_BIN_SUFFIX ,
.Ev GEM_MAN_SUFFIX ,
and and
.Ev GEM_FLAGS . .Ev RUBY .
.Pp .Pp
Appends to Appends to
.Ev CATEGORIES .Ev CATEGORIES
@ -98,23 +91,17 @@ unless
.Ev MODRUBY_RUNDEP .Ev MODRUBY_RUNDEP
is set to No. is set to No.
.Pp .Pp
.Ev MODRUBY_RSPEC_DEPENDS
is set to the dependency line for the rspec 1 port for the version of ruby in
use.
.Ev MODRUBY_RSPEC3_DEPENDS .Ev MODRUBY_RSPEC3_DEPENDS
is set to the dependency line for the rspec 3 port for the version of ruby in is set to the dependency line for the rspec 3 port for the version of ruby in
use. use.
.Ev RUBY , .Ev RUBY ,
.Ev RAKE ,
and
.Ev MODRUBY_BIN_TESTRB
are the path to the ruby, rake, and
testrb binaries for the related ruby implementation.
.Ev RSPEC
and and
.Ev RAKE
are the path to the ruby and rake
binaries for the related ruby implementation.
.Ev MODRUBY_BIN_RSPEC .Ev MODRUBY_BIN_RSPEC
are the path to the spec (rspec 1) and is the path to the rspec (rspec 3) binaries for the related ruby
rspec (rspec 3) binaries for the related ruby implementation. implementation.
.Pp .Pp
.Ev MODRUBY_RUBY_ADJ .Ev MODRUBY_RUBY_ADJ
is a command that takes filename arguments and replaces is a command that takes filename arguments and replaces
@ -129,14 +116,10 @@ Creates a do-test target if
.Ev MODRUBY_TEST .Ev MODRUBY_TEST
is used and one is not already defined. is used and one is not already defined.
.Ev MODRUBY_TEST .Ev MODRUBY_TEST
can be set to ruby, rake, rspec, rspec3, or testrb to can be set to ruby, rake, rspec3, or testrb to
choose the appropriate binary to run the regress tests. choose the appropriate binary to run the regress tests.
.Ev MODRUBY_TEST_ENV .Ev MODRUBY_TEST_ENV
can be used to set environment variables for the regress tests. can be used to set environment variables for the regress tests.
.Ev MODRUBY_TEST_DIR
can be used to set the directory in which to execute
the regress tests, it defaults to
.Ev WRKSRC .
.Ev MODRUBY_TEST_TARGET .Ev MODRUBY_TEST_TARGET
sets the argument to the regress test program. sets the argument to the regress test program.
.Pp .Pp
@ -152,7 +135,7 @@ The "ruby gem"
should be used for pure ruby gems without C extensions. should be used for pure ruby gems without C extensions.
This adds This adds
.Ev PKG_ARCH .Ev PKG_ARCH
= * and adds ruby30, ruby31, ruby32, and jruby = * and adds ruby30, ruby31, and ruby32
.Ev FLAVOR Ns s .Ev FLAVOR Ns s
to the port, so the same port can build packages for multiple versions of ruby. to the port, so the same port can build packages for multiple versions of ruby.
.It .It
@ -192,8 +175,7 @@ instead of ruby as the package prefix.
Specifying Specifying
.Ev MODRUBY_FLAVOR .Ev MODRUBY_FLAVOR
is necessary so that if a ruby x.y package is being built for the current is necessary so that if a ruby x.y package is being built for the current
port, it depends on the ruby x.y package of the port, it depends on the ruby x.y package of the dependencies.
dependencies.
.Pp .Pp
The ports system defaults to using ruby 3.2 if the version of ruby is not The ports system defaults to using ruby 3.2 if the version of ruby is not
specified. specified.
@ -202,7 +184,7 @@ If you want to specify a version for a gem port, use a specific
such as ruby31 to use ruby 3.1. such as ruby31 to use ruby 3.1.
If you want to specify a version for a non-gem port, set If you want to specify a version for a non-gem port, set
.Ev MODRUBY_REV .Ev MODRUBY_REV
for the version of ruby you want to use (3.0, 3.1, 3.2, jruby). for the version of ruby you want to use (3.0, 3.1, 3.2).
.Pp .Pp
If you do not want the ports system to automatically set up If you do not want the ports system to automatically set up
.Ev FLAVOR Ns s .Ev FLAVOR Ns s
@ -218,7 +200,7 @@ When
.Ev MODRUBY_HANDLE_FLAVORS .Ev MODRUBY_HANDLE_FLAVORS
is Yes, the ports system automatically adds the appropriate prefix to the is Yes, the ports system automatically adds the appropriate prefix to the
.Ev FULLPKGNAME .Ev FULLPKGNAME
(e.g. jruby- for jruby, ruby32- for ruby 3.2). (e.g. ruby31- for ruby 3.1, ruby32- for ruby 3.2).
Additionally, it adds Additionally, it adds
.Ev GEM_BIN_SUFFIX .Ev GEM_BIN_SUFFIX
and and

View File

@ -1,7 +1,7 @@
/* $OpenBSD: bt_parse.y,v 1.52 2023/09/02 19:28:46 dv Exp $ */ /* $OpenBSD: bt_parse.y,v 1.53 2023/09/11 19:01:26 mpi Exp $ */
/* /*
* Copyright (c) 2019-2021 Martin Pieuchot <mpi@openbsd.org> * Copyright (c) 2019-2023 Martin Pieuchot <mpi@openbsd.org>
* Copyright (c) 2019 Tobias Heider <tobhe@openbsd.org> * Copyright (c) 2019 Tobias Heider <tobhe@openbsd.org>
* Copyright (c) 2015 Ted Unangst <tedu@openbsd.org> * Copyright (c) 2015 Ted Unangst <tedu@openbsd.org>
* *
@ -72,6 +72,8 @@ struct bt_stmt *bg_store(const char *, struct bt_arg *);
struct bt_arg *bg_find(const char *); struct bt_arg *bg_find(const char *);
struct bt_var *bg_get(const char *); struct bt_var *bg_get(const char *);
struct bt_arg *bi_find(struct bt_arg *, unsigned long);
struct bt_var *bl_lookup(const char *); struct bt_var *bl_lookup(const char *);
struct bt_stmt *bl_store(const char *, struct bt_arg *); struct bt_stmt *bl_store(const char *, struct bt_arg *);
struct bt_arg *bl_find(const char *); struct bt_arg *bl_find(const char *);
@ -210,9 +212,11 @@ fterm : fterm '*' factor { $$ = ba_op(B_AT_OP_MULT, $1, $3); }
variable: lvar { $$ = bl_find($1); } variable: lvar { $$ = bl_find($1); }
| gvar { $$ = bg_find($1); } | gvar { $$ = bg_find($1); }
| variable '.' NUMBER { $$ = bi_find($1, $3); }
; ;
factor : '(' expr ')' { $$ = $2; } factor : '(' expr ')' { $$ = $2; }
| '(' vargs ',' expr ')'{ $$ = ba_new(ba_append($2, $4), B_AT_TUPLE); }
| NUMBER { $$ = ba_new($1, B_AT_LONG); } | NUMBER { $$ = ba_new($1, B_AT_LONG); }
| BUILTIN { $$ = ba_new(NULL, $1); } | BUILTIN { $$ = ba_new(NULL, $1); }
| CSTRING { $$ = ba_new($1, B_AT_STR); } | CSTRING { $$ = ba_new($1, B_AT_STR); }
@ -419,7 +423,7 @@ ba_new0(void *val, enum bt_argtype type)
/* /*
* Link two arguments together, to build an argument list used in * Link two arguments together, to build an argument list used in
* function calls. * operators, tuples and function calls.
*/ */
struct bt_arg * struct bt_arg *
ba_append(struct bt_arg *da0, struct bt_arg *da1) ba_append(struct bt_arg *da0, struct bt_arg *da1)
@ -592,6 +596,17 @@ bl_store(const char *vname, struct bt_arg *vval)
return bs_new(B_AC_STORE, vval, bv); return bs_new(B_AC_STORE, vval, bv);
} }
/* Create an argument that points to a tuple variable and a given index */
struct bt_arg *
bi_find(struct bt_arg *ba, unsigned long index)
{
struct bt_var *bv = ba->ba_value;
ba = ba_new(bv, B_AT_TMEMBER);
ba->ba_key = (void *)index;
return ba;
}
struct bt_stmt * struct bt_stmt *
bm_op(enum bt_action mact, struct bt_arg *ba, struct bt_arg *mval) bm_op(enum bt_action mact, struct bt_arg *ba, struct bt_arg *mval)
{ {
@ -610,7 +625,7 @@ bm_insert(const char *mname, struct bt_arg *mkey, struct bt_arg *mval)
return bs_new(B_AC_INSERT, ba, (struct bt_var *)mval); return bs_new(B_AC_INSERT, ba, (struct bt_var *)mval);
} }
/* Create an argument that points to a variable and attach a key to it. */ /* Create an argument that points to a map variable and attach a key to it. */
struct bt_arg * struct bt_arg *
bm_find(const char *vname, struct bt_arg *mkey) bm_find(const char *vname, struct bt_arg *mkey)
{ {

View File

@ -1,4 +1,4 @@
/* $OpenBSD: bt_parser.h,v 1.23 2022/12/28 21:30:16 jmc Exp $ */ /* $OpenBSD: bt_parser.h,v 1.24 2023/09/11 19:01:26 mpi Exp $ */
/* /*
* Copyright (c) 2019-2021 Martin Pieuchot <mpi@openbsd.org> * Copyright (c) 2019-2021 Martin Pieuchot <mpi@openbsd.org>
@ -94,7 +94,7 @@ struct bt_rule {
}; };
/* /*
* Global variable representation. * Global and local variable representation.
* *
* Variables are untyped and also include maps and histograms. * Variables are untyped and also include maps and histograms.
*/ */
@ -105,6 +105,7 @@ struct bt_var {
enum bt_vartype { enum bt_vartype {
B_VT_STR = 1, B_VT_STR = 1,
B_VT_LONG, B_VT_LONG,
B_VT_TUPLE,
B_VT_MAP, B_VT_MAP,
B_VT_HIST, B_VT_HIST,
} bv_type; } bv_type;
@ -126,6 +127,8 @@ struct bt_arg {
B_AT_VAR, /* global/local variable */ B_AT_VAR, /* global/local variable */
B_AT_MAP, /* global map (@map[]) */ B_AT_MAP, /* global map (@map[]) */
B_AT_HIST, /* histogram */ B_AT_HIST, /* histogram */
B_AT_TUPLE, /* tuple (1, 42, "str") */
B_AT_TMEMBER, /* tuple member $t.2 */
B_AT_NIL, /* empty value */ B_AT_NIL, /* empty value */
B_AT_BI_PID, B_AT_BI_PID,

View File

@ -1,7 +1,7 @@
/* $OpenBSD: btrace.c,v 1.75 2023/09/02 15:16:12 dv Exp $ */ /* $OpenBSD: btrace.c,v 1.76 2023/09/11 19:01:26 mpi Exp $ */
/* /*
* Copyright (c) 2019 - 2021 Martin Pieuchot <mpi@openbsd.org> * Copyright (c) 2019 - 2023 Martin Pieuchot <mpi@openbsd.org>
* *
* Permission to use, copy, modify, and distribute this software for any * Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above * purpose with or without fee is hereby granted, provided that the above
@ -95,6 +95,7 @@ bool stmt_test(struct bt_stmt *, struct dt_evt *);
void stmt_time(struct bt_stmt *, struct dt_evt *); void stmt_time(struct bt_stmt *, struct dt_evt *);
void stmt_zero(struct bt_stmt *); void stmt_zero(struct bt_stmt *);
struct bt_arg *ba_read(struct bt_arg *); struct bt_arg *ba_read(struct bt_arg *);
struct bt_arg *baeval(struct bt_arg *, struct dt_evt *);
const char *ba2hash(struct bt_arg *, struct dt_evt *); const char *ba2hash(struct bt_arg *, struct dt_evt *);
long baexpr2long(struct bt_arg *, struct dt_evt *); long baexpr2long(struct bt_arg *, struct dt_evt *);
const char *ba2bucket(struct bt_arg *, struct bt_arg *, const char *ba2bucket(struct bt_arg *, struct bt_arg *,
@ -854,6 +855,7 @@ stmt_bucketize(struct bt_stmt *bs, struct dt_evt *dtev)
struct bt_arg *brange, *bhist = SLIST_FIRST(&bs->bs_args); struct bt_arg *brange, *bhist = SLIST_FIRST(&bs->bs_args);
struct bt_arg *bval = (struct bt_arg *)bs->bs_var; struct bt_arg *bval = (struct bt_arg *)bs->bs_var;
struct bt_var *bv = bhist->ba_value; struct bt_var *bv = bhist->ba_value;
struct hist *hist;
const char *bucket; const char *bucket;
long step = 0; long step = 0;
@ -870,8 +872,17 @@ stmt_bucketize(struct bt_stmt *bs, struct dt_evt *dtev)
debug("hist=%p '%s' increment bucket '%s'\n", bv->bv_value, debug("hist=%p '%s' increment bucket '%s'\n", bv->bv_value,
bv_name(bv), bucket); bv_name(bv), bucket);
bv->bv_value = (struct bt_arg *) /* hist is NULL before first insert or after clear() */
hist_increment((struct hist *)bv->bv_value, bucket, step); hist = (struct hist *)bv->bv_value;
if (hist == NULL)
hist = hist_new(step);
hist_increment(hist, bucket);
debug("hist=%p '%s' increment bucket=%p '%s' bval=%p\n", hist,
bv_name(bv), brange, bucket, bval);
bv->bv_value = (struct bt_arg *)hist;
bv->bv_type = B_VT_HIST; bv->bv_type = B_VT_HIST;
} }
@ -943,6 +954,7 @@ stmt_insert(struct bt_stmt *bs, struct dt_evt *dtev)
struct bt_var *bv = bmap->ba_value; struct bt_var *bv = bmap->ba_value;
struct map *map; struct map *map;
const char *hash; const char *hash;
long val;
assert(bmap->ba_type == B_AT_MAP); assert(bmap->ba_type == B_AT_MAP);
assert(SLIST_NEXT(bval, ba_next) == NULL); assert(SLIST_NEXT(bval, ba_next) == NULL);
@ -952,7 +964,36 @@ stmt_insert(struct bt_stmt *bs, struct dt_evt *dtev)
/* map is NULL before first insert or after clear() */ /* map is NULL before first insert or after clear() */
map = (struct map *)bv->bv_value; map = (struct map *)bv->bv_value;
map = map_insert(map, hash, bval, dtev); if (map == NULL)
map = map_new();
/* Operate on existring value for count(), max(), min() and sum(). */
switch (bval->ba_type) {
case B_AT_MF_COUNT:
val = ba2long(map_get(map, hash), NULL);
val++;
bval = ba_new(val, B_AT_LONG);
break;
case B_AT_MF_MAX:
val = ba2long(map_get(map, hash), NULL);
val = MAXIMUM(val, ba2long(bval->ba_value, dtev));
bval = ba_new(val, B_AT_LONG);
break;
case B_AT_MF_MIN:
val = ba2long(map_get(map, hash), NULL);
val = MINIMUM(val, ba2long(bval->ba_value, dtev));
bval = ba_new(val, B_AT_LONG);
break;
case B_AT_MF_SUM:
val = ba2long(map_get(map, hash), NULL);
val += ba2long(bval->ba_value, dtev);
bval = ba_new(val, B_AT_LONG);
break;
default:
break;
}
map_insert(map, hash, bval);
debug("map=%p '%s' insert key=%p '%s' bval=%p\n", map, debug("map=%p '%s' insert key=%p '%s' bval=%p\n", map,
bv_name(bv), bkey, hash, bval); bv_name(bv), bkey, hash, bval);
@ -1018,7 +1059,7 @@ void
stmt_store(struct bt_stmt *bs, struct dt_evt *dtev) stmt_store(struct bt_stmt *bs, struct dt_evt *dtev)
{ {
struct bt_arg *ba = SLIST_FIRST(&bs->bs_args); struct bt_arg *ba = SLIST_FIRST(&bs->bs_args);
struct bt_var *bv = bs->bs_var; struct bt_var *bvar, *bv = bs->bs_var;
assert(SLIST_NEXT(ba, ba_next) == NULL); assert(SLIST_NEXT(ba, ba_next) == NULL);
@ -1031,33 +1072,33 @@ stmt_store(struct bt_stmt *bs, struct dt_evt *dtev)
bv->bv_value = ba; bv->bv_value = ba;
bv->bv_type = B_VT_LONG; bv->bv_type = B_VT_LONG;
break; break;
case B_AT_VAR:
bvar = ba->ba_value;
bv->bv_type = bvar->bv_type;
bv->bv_value = bvar->bv_value;
break;
case B_AT_TUPLE:
bv->bv_value = baeval(ba, dtev);
bv->bv_type = B_VT_TUPLE;
break;
case B_AT_BI_PID: case B_AT_BI_PID:
bv->bv_value = ba_new((long)dtev->dtev_pid, B_AT_LONG);
bv->bv_type = B_VT_LONG;
break;
case B_AT_BI_TID: case B_AT_BI_TID:
bv->bv_value = ba_new((long)dtev->dtev_tid, B_AT_LONG);
bv->bv_type = B_VT_LONG;
break;
case B_AT_BI_NSECS: case B_AT_BI_NSECS:
bv->bv_value = ba_new(builtin_nsecs(dtev), B_AT_LONG);
bv->bv_type = B_VT_LONG;
break;
case B_AT_BI_ARG0 ... B_AT_BI_ARG9: case B_AT_BI_ARG0 ... B_AT_BI_ARG9:
/* FALLTHROUGH */
case B_AT_OP_PLUS ... B_AT_OP_LOR: case B_AT_OP_PLUS ... B_AT_OP_LOR:
bv->bv_value = ba_new(ba2long(ba, dtev), B_AT_LONG); bv->bv_value = baeval(ba, dtev);
bv->bv_type = B_VT_LONG; bv->bv_type = B_VT_LONG;
break; break;
case B_AT_FN_STR: case B_AT_FN_STR:
bv->bv_value = ba_new(ba2str(ba, dtev), B_AT_STR); bv->bv_value = baeval(ba, dtev);
bv->bv_type = B_VT_STR; bv->bv_type = B_VT_STR;
break; break;
default: default:
xabort("store not implemented for type %d", ba->ba_type); xabort("store not implemented for type %d", ba->ba_type);
} }
debug("bv=%p var '%s' store (%p)\n", bv, bv_name(bv), bv->bv_value); debug("bv=%p var '%s' store (%p)='%s'\n", bv, bv_name(bv), bv->bv_value,
ba2str(bv->bv_value, dtev));
} }
/* /*
@ -1160,7 +1201,6 @@ ba_read(struct bt_arg *ba)
struct bt_var *bv = ba->ba_value; struct bt_var *bv = ba->ba_value;
assert(ba->ba_type == B_AT_VAR); assert(ba->ba_type == B_AT_VAR);
debug("bv=%p read '%s' (%p)\n", bv, bv_name(bv), bv->bv_value); debug("bv=%p read '%s' (%p)\n", bv, bv_name(bv), bv->bv_value);
/* Handle map/hist access after clear(). */ /* Handle map/hist access after clear(). */
@ -1170,6 +1210,57 @@ ba_read(struct bt_arg *ba)
return bv->bv_value; return bv->bv_value;
} }
// XXX
extern struct bt_arg *ba_append(struct bt_arg *, struct bt_arg *);
/*
* Return a new argument that doesn't depend on `dtev'. This is used
* when storing values in variables, maps, etc.
*/
struct bt_arg *
baeval(struct bt_arg *bval, struct dt_evt *dtev)
{
struct bt_arg *ba, *bh = NULL;
switch (bval->ba_type) {
case B_AT_VAR:
ba = baeval(ba_read(bval), NULL);
break;
case B_AT_LONG:
case B_AT_BI_PID:
case B_AT_BI_TID:
case B_AT_BI_CPU:
case B_AT_BI_NSECS:
case B_AT_BI_ARG0 ... B_AT_BI_ARG9:
case B_AT_BI_RETVAL:
case B_AT_OP_PLUS ... B_AT_OP_LOR:
ba = ba_new(ba2long(bval, dtev), B_AT_LONG);
break;
case B_AT_STR:
case B_AT_BI_COMM:
case B_AT_BI_KSTACK:
case B_AT_BI_USTACK:
case B_AT_BI_PROBE:
case B_AT_FN_STR:
ba = ba_new(ba2str(bval, dtev), B_AT_STR);
break;
case B_AT_TUPLE:
ba = bval->ba_value;
do {
bh = ba_append(bh, baeval(ba, dtev));
} while ((ba = SLIST_NEXT(ba, ba_next)) != NULL);
ba = ba_new(bh, B_AT_TUPLE);
break;
default:
xabort("no eval support for type %d", bval->ba_type);
}
return ba;
}
/*
* Return a string of coma-separated values
*/
const char * const char *
ba2hash(struct bt_arg *ba, struct dt_evt *dtev) ba2hash(struct bt_arg *ba, struct dt_evt *dtev)
{ {
@ -1556,6 +1647,7 @@ ba2str(struct bt_arg *ba, struct dt_evt *dtev)
static char buf[STRLEN]; static char buf[STRLEN];
struct bt_var *bv; struct bt_var *bv;
struct dtioc_probe_info *dtpi; struct dtioc_probe_info *dtpi;
unsigned long idx;
const char *str; const char *str;
buf[0] = '\0'; buf[0] = '\0';
@ -1567,6 +1659,26 @@ ba2str(struct bt_arg *ba, struct dt_evt *dtev)
snprintf(buf, sizeof(buf), "%ld",(long)ba->ba_value); snprintf(buf, sizeof(buf), "%ld",(long)ba->ba_value);
str = buf; str = buf;
break; break;
case B_AT_TUPLE:
snprintf(buf, sizeof(buf), "(%s)", ba2hash(ba->ba_value, dtev));
str = buf;
break;
case B_AT_TMEMBER:
idx = (unsigned long)ba->ba_key;
bv = ba->ba_value;
/* Uninitialized tuple */
if (bv->bv_value == NULL) {
str = buf;
break;
}
ba = bv->bv_value;
assert(ba->ba_type == B_AT_TUPLE);
ba = ba->ba_value;
while (ba != NULL && idx-- > 0) {
ba = SLIST_NEXT(ba, ba_next);
}
str = ba2str(ba, dtev);
break;
case B_AT_NIL: case B_AT_NIL:
str = ""; str = "";
break; break;
@ -1674,6 +1786,8 @@ ba2dtflags(struct bt_arg *ba)
switch (bval->ba_type) { switch (bval->ba_type) {
case B_AT_STR: case B_AT_STR:
case B_AT_LONG: case B_AT_LONG:
case B_AT_TUPLE:
case B_AT_TMEMBER:
case B_AT_VAR: case B_AT_VAR:
case B_AT_HIST: case B_AT_HIST:
case B_AT_NIL: case B_AT_NIL:
@ -1720,6 +1834,8 @@ ba2dtflags(struct bt_arg *ba)
long long
bacmp(struct bt_arg *a, struct bt_arg *b) bacmp(struct bt_arg *a, struct bt_arg *b)
{ {
long val;
if (a->ba_type != b->ba_type) if (a->ba_type != b->ba_type)
return a->ba_type - b->ba_type; return a->ba_type - b->ba_type;
@ -1728,8 +1844,24 @@ bacmp(struct bt_arg *a, struct bt_arg *b)
return ba2long(a, NULL) - ba2long(b, NULL); return ba2long(a, NULL) - ba2long(b, NULL);
case B_AT_STR: case B_AT_STR:
return strcmp(ba2str(a, NULL), ba2str(b, NULL)); return strcmp(ba2str(a, NULL), ba2str(b, NULL));
case B_AT_TUPLE:
/* Compare two lists of arguments one by one. */
do {
val = bacmp(a, b);
if (val != 0)
break;
a = SLIST_NEXT(a, ba_next);
b = SLIST_NEXT(b, ba_next);
if (a == NULL && b != NULL)
val = -1;
else if (a != NULL && b == NULL)
val = 1;
} while (a != NULL && b != NULL);
return val;
default: default:
errx(1, "no compare support for type %d", a->ba_type); xabort("no compare support for type %d", a->ba_type);
} }
} }

View File

@ -1,4 +1,4 @@
/* $OpenBSD: btrace.h,v 1.12 2023/05/12 14:14:16 claudio Exp $ */ /* $OpenBSD: btrace.h,v 1.13 2023/09/11 19:01:26 mpi Exp $ */
/* /*
* Copyright (c) 2019 - 2020 Martin Pieuchot <mpi@openbsd.org> * Copyright (c) 2019 - 2020 Martin Pieuchot <mpi@openbsd.org>
@ -46,14 +46,15 @@ int kelf_snprintsym(struct syms *, char *, size_t,
/* map.c */ /* map.c */
struct map; struct map;
struct hist; struct hist;
struct map *map_new(void);
void map_clear(struct map *); void map_clear(struct map *);
void map_delete(struct map *, const char *); void map_delete(struct map *, const char *);
struct bt_arg *map_get(struct map *, const char *); struct bt_arg *map_get(struct map *, const char *);
struct map *map_insert(struct map *, const char *, struct bt_arg *, void map_insert(struct map *, const char *, void *);
struct dt_evt *);
void map_print(struct map *, size_t, const char *); void map_print(struct map *, size_t, const char *);
void map_zero(struct map *); void map_zero(struct map *);
struct hist *hist_increment(struct hist *, const char *, long); struct hist *hist_new(long);
void hist_increment(struct hist *, const char *);
void hist_print(struct hist *, const char *); void hist_print(struct hist *, const char *);
#define KLEN 1024 /* # of characters in map key, contain a stack trace */ #define KLEN 1024 /* # of characters in map key, contain a stack trace */

View File

@ -1,4 +1,4 @@
/* $OpenBSD: map.c,v 1.23 2023/09/03 10:26:35 mpi Exp $ */ /* $OpenBSD: map.c,v 1.24 2023/09/11 19:01:26 mpi Exp $ */
/* /*
* Copyright (c) 2020 Martin Pieuchot <mpi@openbsd.org> * Copyright (c) 2020 Martin Pieuchot <mpi@openbsd.org>
@ -33,14 +33,6 @@
#include "bt_parser.h" #include "bt_parser.h"
#include "btrace.h" #include "btrace.h"
#ifndef MIN
#define MIN(_a,_b) ((_a) < (_b) ? (_a) : (_b))
#endif
#ifndef MAX
#define MAX(_a,_b) ((_a) > (_b) ? (_a) : (_b))
#endif
RB_HEAD(map, mentry); RB_HEAD(map, mentry);
struct mentry { struct mentry {
@ -79,6 +71,18 @@ mget(struct map *map, const char *key)
return mep; return mep;
} }
struct map *
map_new(void)
{
struct map *map;
map = calloc(1, sizeof(struct map));
if (map == NULL)
err(1, "map: calloc");
return map;
}
void void
map_clear(struct map *map) map_clear(struct map *map)
{ {
@ -118,75 +122,14 @@ map_get(struct map *map, const char *key)
return mep->mval; return mep->mval;
} }
struct map * void
map_insert(struct map *map, const char *key, struct bt_arg *bval, map_insert(struct map *map, const char *key, void *cookie)
struct dt_evt *dtev)
{ {
struct mentry *mep; struct mentry *mep;
long val;
if (map == NULL) {
map = calloc(1, sizeof(struct map));
if (map == NULL)
err(1, "map: calloc");
}
mep = mget(map, key); mep = mget(map, key);
switch (bval->ba_type) {
case B_AT_STR:
free(mep->mval); free(mep->mval);
mep->mval = ba_new(ba2str(bval, dtev), B_AT_LONG); mep->mval = cookie;
break;
case B_AT_LONG:
case B_AT_BI_PID:
case B_AT_BI_TID:
case B_AT_BI_CPU:
case B_AT_BI_NSECS:
case B_AT_BI_ARG0 ... B_AT_BI_ARG9:
case B_AT_BI_RETVAL:
case B_AT_BI_PROBE:
free(mep->mval);
mep->mval = ba_new(ba2long(bval, dtev), B_AT_LONG);
break;
case B_AT_MF_COUNT:
if (mep->mval == NULL)
mep->mval = ba_new(0, B_AT_LONG);
val = (long)mep->mval->ba_value;
val++;
mep->mval->ba_value = (void *)val;
break;
case B_AT_MF_MAX:
if (mep->mval == NULL)
mep->mval = ba_new(0, B_AT_LONG);
val = (long)mep->mval->ba_value;
val = MAX(val, ba2long(bval->ba_value, dtev));
mep->mval->ba_value = (void *)val;
break;
case B_AT_MF_MIN:
if (mep->mval == NULL)
mep->mval = ba_new(0, B_AT_LONG);
val = (long)mep->mval->ba_value;
val = MIN(val, ba2long(bval->ba_value, dtev));
mep->mval->ba_value = (void *)val;
break;
case B_AT_MF_SUM:
if (mep->mval == NULL)
mep->mval = ba_new(0, B_AT_LONG);
val = (long)mep->mval->ba_value;
val += ba2long(bval->ba_value, dtev);
mep->mval->ba_value = (void *)val;
break;
case B_AT_BI_COMM:
case B_AT_BI_KSTACK:
case B_AT_BI_USTACK:
free(mep->mval);
mep->mval = ba_new(ba2str(bval, dtev), B_AT_STR);
break;
default:
errx(1, "no insert support for type %d", bval->ba_type);
}
return map;
} }
static int static int
@ -248,26 +191,36 @@ map_zero(struct map *map)
/* /*
* Histogram implemented with map. * Histogram implemented with map.
*/ */
struct hist { struct hist {
struct map hmap; struct map hmap;
int hstep; int hstep;
}; };
struct hist * struct hist *
hist_increment(struct hist *hist, const char *key, long step) hist_new(long step)
{ {
static struct bt_arg incba = BA_INITIALIZER(NULL, B_AT_MF_COUNT); struct hist *hist;
if (hist == NULL) {
hist = calloc(1, sizeof(struct hist)); hist = calloc(1, sizeof(struct hist));
if (hist == NULL) if (hist == NULL)
err(1, "hist: calloc"); err(1, "hist: calloc");
hist->hstep = step; hist->hstep = step;
}
assert(hist->hstep == step);
return (struct hist *)map_insert(&hist->hmap, key, &incba, NULL); return hist;
}
void
hist_increment(struct hist *hist, const char *bucket)
{
struct bt_arg *ba;
long val;
ba = map_get(&hist->hmap, bucket);
assert(ba->ba_type == B_AT_LONG);
val = (long)ba->ba_value;
val++;
ba->ba_value = (void *)val;
} }
long long