| /* sysprof-capture-condition.c |
| * |
| * Copyright 2016-2019 Christian Hergert <[email protected]> |
| * |
| * 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. |
| * |
| * 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. |
| * |
| * Subject to the terms and conditions of this license, each copyright holder |
| * and contributor hereby grants to those receiving rights under this license |
| * a perpetual, worldwide, non-exclusive, no-charge, royalty-free, |
| * irrevocable (except for failure to satisfy the conditions of this license) |
| * patent license to make, have made, use, offer to sell, sell, import, and |
| * otherwise transfer this software, where such license applies only to those |
| * patent claims, already acquired or hereafter acquired, licensable by such |
| * copyright holder or contributor that are necessarily infringed by: |
| * |
| * (a) their Contribution(s) (the licensed copyrights of copyright holders |
| * and non-copyrightable additions of contributors, in source or binary |
| * form) alone; or |
| * |
| * (b) combination of their Contribution(s) with the work of authorship to |
| * which such Contribution(s) was added by such copyright holder or |
| * contributor, if, at the time the Contribution is added, such addition |
| * causes such combination to be necessarily infringed. The patent license |
| * shall not apply to any other combinations which include the |
| * Contribution. |
| * |
| * Except as expressly stated above, no rights or licenses from any copyright |
| * holder or contributor is granted under this license, whether expressly, by |
| * implication, estoppel or otherwise. |
| * |
| * DISCLAIMER |
| * |
| * 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 MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR |
| * CONTRIBUTORS 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. |
| * |
| * SPDX-License-Identifier: BSD-2-Clause-Patent |
| */ |
| |
| #include "config.h" |
| |
| #include <assert.h> |
| #include <stdbool.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #include "sysprof-capture-condition.h" |
| #include "sysprof-capture-util-private.h" |
| #include "sysprof-macros-internal.h" |
| |
| /** |
| * SECTION:sysprof-capture-condition |
| * @title: SysprofCaptureCondition |
| * |
| * The #SysprofCaptureCondition type is an abstraction on an operation |
| * for a sort of AST to the #SysprofCaptureCursor. The goal is that if |
| * we abstract the types of fields we want to match in the cursor |
| * that we can opportunistically add indexes to speed up the operation |
| * later on without changing the implementation of cursor consumers. |
| */ |
| |
| typedef enum |
| { |
| SYSPROF_CAPTURE_CONDITION_AND, |
| SYSPROF_CAPTURE_CONDITION_OR, |
| SYSPROF_CAPTURE_CONDITION_WHERE_TYPE_IN, |
| SYSPROF_CAPTURE_CONDITION_WHERE_TIME_BETWEEN, |
| SYSPROF_CAPTURE_CONDITION_WHERE_PID_IN, |
| SYSPROF_CAPTURE_CONDITION_WHERE_COUNTER_IN, |
| SYSPROF_CAPTURE_CONDITION_WHERE_FILE, |
| } SysprofCaptureConditionType; |
| |
| struct _SysprofCaptureCondition |
| { |
| volatile int ref_count; |
| SysprofCaptureConditionType type; |
| union { |
| struct { |
| SysprofCaptureFrameType *data; |
| size_t len; |
| } where_type_in; |
| struct { |
| int64_t begin; |
| int64_t end; |
| } where_time_between; |
| struct { |
| int32_t *data; |
| size_t len; |
| } where_pid_in; |
| struct { |
| unsigned int *data; |
| size_t len; |
| } where_counter_in; |
| struct { |
| SysprofCaptureCondition *left; |
| SysprofCaptureCondition *right; |
| } and, or; |
| char *where_file; |
| } u; |
| }; |
| |
| bool |
| sysprof_capture_condition_match (const SysprofCaptureCondition *self, |
| const SysprofCaptureFrame *frame) |
| { |
| assert (self != NULL); |
| assert (frame != NULL); |
| |
| switch (self->type) |
| { |
| case SYSPROF_CAPTURE_CONDITION_AND: |
| return sysprof_capture_condition_match (self->u.and.left, frame) && |
| sysprof_capture_condition_match (self->u.and.right, frame); |
| |
| case SYSPROF_CAPTURE_CONDITION_OR: |
| return sysprof_capture_condition_match (self->u.or.left, frame) || |
| sysprof_capture_condition_match (self->u.or.right, frame); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_TYPE_IN: |
| for (size_t i = 0; i < self->u.where_type_in.len; i++) |
| { |
| if (frame->type == self->u.where_type_in.data[i]) |
| return true; |
| } |
| return false; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_TIME_BETWEEN: |
| return (frame->time >= self->u.where_time_between.begin && frame->time <= self->u.where_time_between.end); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_PID_IN: |
| for (size_t i = 0; i < self->u.where_pid_in.len; i++) |
| { |
| if (frame->pid == self->u.where_pid_in.data[i]) |
| return true; |
| } |
| return false; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_COUNTER_IN: |
| if (frame->type == SYSPROF_CAPTURE_FRAME_CTRSET) |
| { |
| const SysprofCaptureCounterSet *set = (SysprofCaptureCounterSet *)frame; |
| |
| for (size_t i = 0; i < self->u.where_counter_in.len; i++) |
| { |
| unsigned int counter = self->u.where_counter_in.data[i]; |
| |
| for (unsigned int j = 0; j < set->n_values; j++) |
| { |
| if (counter == set->values[j].ids[0] || |
| counter == set->values[j].ids[1] || |
| counter == set->values[j].ids[2] || |
| counter == set->values[j].ids[3] || |
| counter == set->values[j].ids[4] || |
| counter == set->values[j].ids[5] || |
| counter == set->values[j].ids[6] || |
| counter == set->values[j].ids[7]) |
| return true; |
| } |
| } |
| } |
| else if (frame->type == SYSPROF_CAPTURE_FRAME_CTRDEF) |
| { |
| const SysprofCaptureCounterDefine *def = (SysprofCaptureCounterDefine *)frame; |
| |
| for (size_t i = 0; i < self->u.where_counter_in.len; i++) |
| { |
| unsigned int counter = self->u.where_counter_in.data[i]; |
| |
| for (unsigned int j = 0; j < def->n_counters; j++) |
| { |
| if (def->counters[j].id == counter) |
| return true; |
| } |
| } |
| } |
| |
| return false; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_FILE: |
| if (frame->type != SYSPROF_CAPTURE_FRAME_FILE_CHUNK) |
| return false; |
| |
| if (self->u.where_file == NULL) |
| return false; |
| |
| return strcmp (((const SysprofCaptureFileChunk *)frame)->path, self->u.where_file) == 0; |
| |
| default: |
| break; |
| } |
| |
| sysprof_assert_not_reached (); |
| |
| return false; |
| } |
| |
| static SysprofCaptureCondition * |
| sysprof_capture_condition_init (void) |
| { |
| SysprofCaptureCondition *self; |
| |
| self = sysprof_malloc0 (sizeof (SysprofCaptureCondition)); |
| if (self == NULL) |
| return NULL; |
| |
| self->ref_count = 1; |
| |
| return sysprof_steal_pointer (&self); |
| } |
| |
| /* Returns NULL on allocation failure. */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_copy (const SysprofCaptureCondition *self) |
| { |
| switch (self->type) |
| { |
| case SYSPROF_CAPTURE_CONDITION_AND: |
| return sysprof_capture_condition_new_and ( |
| sysprof_capture_condition_copy (self->u.and.left), |
| sysprof_capture_condition_copy (self->u.and.right)); |
| |
| case SYSPROF_CAPTURE_CONDITION_OR: |
| return sysprof_capture_condition_new_or ( |
| sysprof_capture_condition_copy (self->u.or.left), |
| sysprof_capture_condition_copy (self->u.or.right)); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_TYPE_IN: |
| return sysprof_capture_condition_new_where_type_in ( |
| self->u.where_type_in.len, |
| self->u.where_type_in.data); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_TIME_BETWEEN: |
| return sysprof_capture_condition_new_where_time_between ( |
| self->u.where_time_between.begin, |
| self->u.where_time_between.end); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_PID_IN: |
| return sysprof_capture_condition_new_where_pid_in ( |
| self->u.where_pid_in.len, |
| self->u.where_pid_in.data); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_COUNTER_IN: |
| return sysprof_capture_condition_new_where_counter_in ( |
| self->u.where_counter_in.len, |
| self->u.where_counter_in.data); |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_FILE: |
| return sysprof_capture_condition_new_where_file (self->u.where_file); |
| |
| default: |
| break; |
| } |
| |
| sysprof_assert_not_reached (); |
| return NULL; |
| } |
| |
| static void |
| sysprof_capture_condition_finalize (SysprofCaptureCondition *self) |
| { |
| switch (self->type) |
| { |
| case SYSPROF_CAPTURE_CONDITION_AND: |
| sysprof_capture_condition_unref (self->u.and.left); |
| sysprof_capture_condition_unref (self->u.and.right); |
| break; |
| |
| case SYSPROF_CAPTURE_CONDITION_OR: |
| sysprof_capture_condition_unref (self->u.or.left); |
| sysprof_capture_condition_unref (self->u.or.right); |
| break; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_TYPE_IN: |
| free (self->u.where_type_in.data); |
| break; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_TIME_BETWEEN: |
| break; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_PID_IN: |
| free (self->u.where_pid_in.data); |
| break; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_COUNTER_IN: |
| free (self->u.where_counter_in.data); |
| break; |
| |
| case SYSPROF_CAPTURE_CONDITION_WHERE_FILE: |
| free (self->u.where_file); |
| break; |
| |
| default: |
| sysprof_assert_not_reached (); |
| break; |
| } |
| |
| free (self); |
| } |
| |
| SysprofCaptureCondition * |
| sysprof_capture_condition_ref (SysprofCaptureCondition *self) |
| { |
| assert (self != NULL); |
| assert (self->ref_count > 0); |
| |
| __atomic_fetch_add (&self->ref_count, 1, __ATOMIC_SEQ_CST); |
| return self; |
| } |
| |
| void |
| sysprof_capture_condition_unref (SysprofCaptureCondition *self) |
| { |
| assert (self != NULL); |
| assert (self->ref_count > 0); |
| |
| if (__atomic_fetch_sub (&self->ref_count, 1, __ATOMIC_SEQ_CST) == 1) |
| sysprof_capture_condition_finalize (self); |
| } |
| |
| /* Returns NULL on allocation failure. */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_where_type_in (unsigned int n_types, |
| const SysprofCaptureFrameType *types) |
| { |
| SysprofCaptureCondition *self; |
| |
| assert (types != NULL); |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_WHERE_TYPE_IN; |
| self->u.where_type_in.data = calloc (n_types, sizeof (SysprofCaptureFrameType)); |
| if (self->u.where_type_in.data == NULL) |
| return NULL; |
| self->u.where_type_in.len = n_types; |
| memcpy (self->u.where_type_in.data, types, sizeof (SysprofCaptureFrameType) * n_types); |
| |
| return self; |
| } |
| |
| /* Returns NULL on allocation failure. */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_where_time_between (int64_t begin_time, |
| int64_t end_time) |
| { |
| SysprofCaptureCondition *self; |
| |
| if SYSPROF_UNLIKELY (begin_time > end_time) |
| { |
| int64_t tmp = begin_time; |
| begin_time = end_time; |
| end_time = tmp; |
| } |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_WHERE_TIME_BETWEEN; |
| self->u.where_time_between.begin = begin_time; |
| self->u.where_time_between.end = end_time; |
| |
| return self; |
| } |
| |
| /* Returns NULL on allocation failure. */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_where_pid_in (unsigned int n_pids, |
| const int32_t *pids) |
| { |
| SysprofCaptureCondition *self; |
| |
| assert (pids != NULL); |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_WHERE_PID_IN; |
| self->u.where_pid_in.data = calloc (n_pids, sizeof (int32_t)); |
| if (self->u.where_pid_in.data == NULL) |
| { |
| free (self); |
| return NULL; |
| } |
| self->u.where_pid_in.len = n_pids; |
| memcpy (self->u.where_pid_in.data, pids, sizeof (int32_t) * n_pids); |
| |
| return self; |
| } |
| |
| /* Returns NULL on allocation failure. */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_where_counter_in (unsigned int n_counters, |
| const unsigned int *counters) |
| { |
| SysprofCaptureCondition *self; |
| |
| assert (counters != NULL || n_counters == 0); |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_WHERE_COUNTER_IN; |
| self->u.where_counter_in.data = calloc (n_counters, sizeof (unsigned int)); |
| if (n_counters > 0 && self->u.where_counter_in.data == NULL) |
| { |
| free (self); |
| return NULL; |
| } |
| self->u.where_counter_in.len = n_counters; |
| |
| if (n_counters > 0) |
| memcpy (self->u.where_counter_in.data, counters, sizeof (unsigned int) * n_counters); |
| |
| return self; |
| } |
| |
| /** |
| * sysprof_capture_condition_new_and: |
| * @left: (transfer full): An #SysprofCaptureCondition |
| * @right: (transfer full): An #SysprofCaptureCondition |
| * |
| * Creates a new #SysprofCaptureCondition that requires both left and right |
| * to evaluate to %TRUE. |
| * |
| * Returns: (transfer full) (nullable): A new #SysprofCaptureCondition, or %NULL |
| * on allocation failure. |
| */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_and (SysprofCaptureCondition *left, |
| SysprofCaptureCondition *right) |
| { |
| SysprofCaptureCondition *self; |
| |
| assert (left != NULL); |
| assert (right != NULL); |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_AND; |
| self->u.and.left = left; |
| self->u.and.right = right; |
| |
| return self; |
| } |
| |
| /** |
| * sysprof_capture_condition_new_or: |
| * @left: (transfer full): An #SysprofCaptureCondition |
| * @right: (transfer full): An #SysprofCaptureCondition |
| * |
| * Creates a new #SysprofCaptureCondition that requires either left and right |
| * to evaluate to %TRUE. |
| * |
| * Returns: (transfer full) (nullable): A new #SysprofCaptureCondition, or %NULL |
| * on allocation failure. |
| */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_or (SysprofCaptureCondition *left, |
| SysprofCaptureCondition *right) |
| { |
| SysprofCaptureCondition *self; |
| |
| assert (left != NULL); |
| assert (right != NULL); |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_OR; |
| self->u.or.left = left; |
| self->u.or.right = right; |
| |
| return self; |
| } |
| |
| /** |
| * sysprof_capture_condition_new_where_file: |
| * @path: a file path to lookup |
| * |
| * Creates a new condition that matches #SysprofCaptureFileChunk frames |
| * which contain the path @path. |
| * |
| * Returns: (transfer full) (nullable): a new #SysprofCaptureCondition, or %NULL |
| * on allocation failure. |
| */ |
| SysprofCaptureCondition * |
| sysprof_capture_condition_new_where_file (const char *path) |
| { |
| SysprofCaptureCondition *self; |
| |
| assert (path != NULL); |
| |
| self = sysprof_capture_condition_init (); |
| if (self == NULL) |
| return NULL; |
| |
| self->type = SYSPROF_CAPTURE_CONDITION_WHERE_FILE; |
| self->u.where_file = sysprof_strdup (path); |
| if (self->u.where_file == NULL) |
| { |
| free (self); |
| return NULL; |
| } |
| |
| return self; |
| } |