/* * * Conky, a system monitor, based on torsmo * * Any original torsmo code is licensed under the BSD license * * All code written since the fork of torsmo is licensed under the GPL * * Please see COPYING for details * * Copyright (c) 2004, Hannu Saransaari and Lauri Hakkarainen * Copyright (c) 2005-2021 Brenden Matthews, Philip Kovacs, et. al. * (see AUTHORS) * 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 3 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, see . * */ #include #include "core.h" #include "logging.h" #include "text_object.h" struct combine_data { char *left; char *seperation; char *right; }; void parse_combine_arg(struct text_object *obj, const char *arg) { struct combine_data *cd; unsigned int i, j; unsigned int indenting = 0; // vars can be used as args for other vars int startvar[2]; int endvar[2]; startvar[0] = endvar[0] = startvar[1] = endvar[1] = -1; j = 0; for (i = 0; arg[i] != 0 && j < 2; i++) { if (startvar[j] == -1) { if (arg[i] == '$') { startvar[j] = i; } } else if (endvar[j] == -1) { if (arg[i] == '{') { indenting++; } else if (arg[i] == '}') { indenting--; } if (indenting == 0 && arg[i + 1] < 48) { //<48 has 0, $, and the most used chars not used // in varnames but not { or } endvar[j] = i + 1; j++; } } } if (startvar[0] >= 0 && endvar[0] >= 0 && startvar[1] >= 0 && endvar[1] >= 0) { cd = static_cast(malloc(sizeof(struct combine_data))); memset(cd, 0, sizeof(struct combine_data)); cd->left = static_cast(malloc(endvar[0] - startvar[0] + 1)); cd->seperation = static_cast(malloc(startvar[1] - endvar[0] + 1)); cd->right = static_cast(malloc(endvar[1] - startvar[1] + 1)); strncpy(cd->left, arg + startvar[0], endvar[0] - startvar[0]); cd->left[endvar[0] - startvar[0]] = 0; strncpy(cd->seperation, arg + endvar[0], startvar[1] - endvar[0]); cd->seperation[startvar[1] - endvar[0]] = 0; strncpy(cd->right, arg + startvar[1], endvar[1] - startvar[1]); cd->right[endvar[1] - startvar[1]] = 0; obj->sub = static_cast(malloc(sizeof(struct text_object))); extract_variable_text_internal(obj->sub, cd->left); obj->sub->sub = static_cast(malloc(sizeof(struct text_object))); extract_variable_text_internal(obj->sub->sub, cd->right); obj->data.opaque = cd; } else { throw combine_needs_2_args_error(); } } void print_combine(struct text_object *obj, char *p, unsigned int p_max_size) { auto *cd = static_cast(obj->data.opaque); std::vector> buf; buf.resize(2); buf[0].resize(max_user_text.get(*state)); buf[1].resize(max_user_text.get(*state)); int i, j; int p_len_remaining = p_max_size - 1; long longest = 0; int nextstart; int nr_rows[2]; struct llrows { char *row; struct llrows *next; }; struct llrows *ll_rows[2], *current[2]; struct text_object *objsub = obj->sub; if ((cd == nullptr) || (p_max_size == 0)) { return; } p[0] = 0; for (i = 0; i < 2; i++) { nr_rows[i] = 1; nextstart = 0; ll_rows[i] = static_cast(malloc(sizeof(struct llrows))); current[i] = ll_rows[i]; for (j = 0; j < i; j++) { objsub = objsub->sub; } generate_text_internal(&(buf[i][0]), max_user_text.get(*state), *objsub); for (j = 0; buf[i][j] != 0; j++) { if (buf[i][j] == '\t') { buf[i][j] = ' '; } if (buf[i][j] == '\n') { buf[i][j] = 0; // the vars inside combine may not have a \n at the end } if (buf[i][j] == 2) { // \002 is used instead of \n to separate lines inside a var buf[i][j] = 0; current[i]->row = strdup(&(buf[i][0]) + nextstart); if (i == 0 && static_cast(strlen(current[i]->row)) > longest) { longest = static_cast(strlen(current[i]->row)); } current[i]->next = static_cast(malloc(sizeof(struct llrows))); current[i] = current[i]->next; nextstart = j + 1; nr_rows[i]++; } } current[i]->row = strdup(&(buf[i][0]) + nextstart); if (i == 0 && static_cast(strlen(current[i]->row)) > longest) { longest = static_cast(strlen(current[i]->row)); } current[i]->next = nullptr; current[i] = ll_rows[i]; } for (j = 0; j < (nr_rows[0] > nr_rows[1] ? nr_rows[0] : nr_rows[1]); j++) { if (current[0] != nullptr) { strncat(p, current[0]->row, p_len_remaining); p_len_remaining -= strlen(current[0]->row); i = strlen(current[0]->row); } else { i = 0; } while (i < longest) { strncat(p, " ", p_len_remaining); p_len_remaining -= 2; i++; } if (current[1] != nullptr) { p_len_remaining -= strlen(cd->seperation); strncat(p, cd->seperation, p_len_remaining); p_len_remaining -= strlen(current[1]->row); strncat(p, current[1]->row, p_len_remaining); } strncat(p, "\n", p_len_remaining); p_len_remaining -= 2; #ifdef HAVE_OPENMP #pragma omp parallel for schedule(dynamic, 10) #endif /* HAVE_OPENMP */ for (i = 0; i < 2; i++) { if (current[i] != nullptr) { current[i] = current[i]->next; } } } #ifdef HAVE_OPENMP #pragma omp parallel for schedule(dynamic, 10) #endif /* HAVE_OPENMP */ for (i = 0; i < 2; i++) { while (ll_rows[i] != nullptr) { current[i] = ll_rows[i]; free(current[i]->row); ll_rows[i] = current[i]->next; free(current[i]); } } } void free_combine(struct text_object *obj) { auto *cd = static_cast(obj->data.opaque); if (cd == nullptr) { return; } free(cd->left); free(cd->seperation); free(cd->right); free_text_objects(obj->sub->sub); free_and_zero(obj->sub->sub); free_text_objects(obj->sub); free_and_zero(obj->sub); free_and_zero(obj->data.opaque); }