2018-05-12 16:03:00 +00:00
|
|
|
/*
|
2009-10-15 19:51:21 +00:00
|
|
|
*
|
|
|
|
* 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
|
2019-01-05 15:52:43 +00:00
|
|
|
* Copyright (c) 2005-2019 Brenden Matthews, Philip Kovacs, et. al.
|
2009-10-15 19:51:21 +00:00
|
|
|
* (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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-23 19:40:34 +00:00
|
|
|
#include "exec.h"
|
2018-05-12 23:26:31 +00:00
|
|
|
#include <fcntl.h>
|
2018-05-12 16:03:00 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <cmath>
|
2018-05-12 23:26:31 +00:00
|
|
|
#include <cstdio>
|
2018-05-12 16:03:00 +00:00
|
|
|
#include <mutex>
|
2009-10-15 19:51:21 +00:00
|
|
|
#include "conky.h"
|
|
|
|
#include "core.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "specials.h"
|
|
|
|
#include "text_object.h"
|
2016-01-09 17:19:48 +00:00
|
|
|
#include "update-cb.hh"
|
2009-10-15 19:51:21 +00:00
|
|
|
|
2009-10-06 22:09:14 +00:00
|
|
|
struct execi_data {
|
2018-05-12 23:26:31 +00:00
|
|
|
float interval{0};
|
|
|
|
char *cmd{nullptr};
|
|
|
|
execi_data() = default;
|
2009-10-06 22:09:14 +00:00
|
|
|
};
|
|
|
|
|
2018-08-09 12:01:26 +00:00
|
|
|
static const int cmd_len = 256;
|
|
|
|
static char cmd[cmd_len];
|
2018-08-08 14:38:32 +00:00
|
|
|
|
2018-08-09 12:01:26 +00:00
|
|
|
static char *remove_excess_quotes(const char *);
|
|
|
|
static char *remove_excess_quotes(const char *command) {
|
|
|
|
char *cmd_ptr = cmd;
|
|
|
|
const char *command_ptr = command;
|
2018-08-08 12:51:35 +00:00
|
|
|
int skip = 0;
|
|
|
|
|
2018-08-09 12:01:26 +00:00
|
|
|
if ((cmd_len - 1) < (strlen(command) - 1)) {
|
|
|
|
snprintf(cmd, cmd_len - 1, "%s", command);
|
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*command_ptr == '"' || *command_ptr == '\'') {
|
|
|
|
skip = 1;
|
|
|
|
command_ptr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; *command_ptr; command_ptr++) {
|
2019-02-23 19:40:34 +00:00
|
|
|
if ('\0' == *(command_ptr + 1) && 1 == skip &&
|
2018-08-09 12:01:26 +00:00
|
|
|
(*command_ptr == '"' || *command_ptr == '\'')) {
|
|
|
|
continue;
|
2018-08-08 13:14:59 +00:00
|
|
|
}
|
2018-08-09 12:01:26 +00:00
|
|
|
*cmd_ptr++ = *command_ptr;
|
2018-08-08 12:51:35 +00:00
|
|
|
}
|
2018-08-09 12:01:26 +00:00
|
|
|
*cmd_ptr = '\0';
|
2018-08-08 14:38:32 +00:00
|
|
|
return cmd;
|
|
|
|
}
|
2018-08-08 12:51:35 +00:00
|
|
|
|
2018-08-08 14:38:32 +00:00
|
|
|
// our own implementation of popen, the difference : the value of 'childpid'
|
|
|
|
// will be filled with the pid of the running 'command'. This is useful if want
|
|
|
|
// to kill it when it hangs while reading or writing to it. We have to kill it
|
|
|
|
// because pclose will wait until the process dies by itself
|
|
|
|
static FILE *pid_popen(const char *command, const char *mode, pid_t *child) {
|
|
|
|
int ends[2];
|
|
|
|
int parentend, childend;
|
2018-05-12 16:03:00 +00:00
|
|
|
|
|
|
|
// by running pipe after the strcmp's we make sure that we don't have to
|
|
|
|
// create a pipe and close the ends if mode is something illegal
|
|
|
|
if (strcmp(mode, "r") == 0) {
|
2018-05-25 00:24:09 +00:00
|
|
|
if (pipe(ends) != 0) { return nullptr; }
|
2018-05-12 16:03:00 +00:00
|
|
|
parentend = ends[0];
|
|
|
|
childend = ends[1];
|
|
|
|
} else if (strcmp(mode, "w") == 0) {
|
2018-05-25 00:24:09 +00:00
|
|
|
if (pipe(ends) != 0) { return nullptr; }
|
2018-05-12 16:03:00 +00:00
|
|
|
parentend = ends[1];
|
|
|
|
childend = ends[0];
|
|
|
|
} else {
|
2018-05-12 23:26:31 +00:00
|
|
|
return nullptr;
|
2018-05-12 16:03:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
*child = fork();
|
|
|
|
if (*child == -1) {
|
|
|
|
close(parentend);
|
|
|
|
close(childend);
|
2018-05-12 23:26:31 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (*child > 0) {
|
2018-05-12 16:03:00 +00:00
|
|
|
close(childend);
|
2018-05-12 23:26:31 +00:00
|
|
|
waitpid(*child, nullptr, 0);
|
2018-05-12 16:03:00 +00:00
|
|
|
} else {
|
|
|
|
// don't read from both stdin and pipe or write to both stdout and pipe
|
|
|
|
if (childend == ends[0]) {
|
|
|
|
close(0);
|
|
|
|
} else {
|
|
|
|
close(1);
|
|
|
|
}
|
|
|
|
close(parentend);
|
|
|
|
|
|
|
|
// by dupping childend, the returned fd will have close-on-exec turned off
|
2018-05-25 00:24:09 +00:00
|
|
|
if (fcntl(childend, F_DUPFD, 0) == -1) { perror("fcntl()"); }
|
2018-05-12 16:03:00 +00:00
|
|
|
close(childend);
|
|
|
|
|
2019-02-23 19:40:34 +00:00
|
|
|
execl("/bin/sh", "sh", "-c", remove_excess_quotes(command),
|
|
|
|
(char *)nullptr);
|
2018-05-12 16:03:00 +00:00
|
|
|
_exit(EXIT_FAILURE); // child should die here, (normally execl will take
|
|
|
|
// care of this but it can fail)
|
|
|
|
}
|
|
|
|
|
|
|
|
return fdopen(parentend, mode);
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Executes a command and stores the result
|
|
|
|
*
|
|
|
|
* This function is called automatically, either once every update
|
|
|
|
* interval, or at specific intervals in the case of execi commands.
|
|
|
|
* conky::run_all_callbacks() handles this. In order for this magic to
|
|
|
|
* happen, we must register a callback with conky::register_cb<exec_cb>()
|
|
|
|
* and store it somewhere, such as obj->exec_handle. To retrieve the
|
|
|
|
* results, use the stored callback to call get_result_copy(), which
|
|
|
|
* returns a std::string.
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void exec_cb::work() {
|
|
|
|
pid_t childpid;
|
|
|
|
std::string buf;
|
|
|
|
std::shared_ptr<FILE> fp;
|
|
|
|
char b[0x1000];
|
|
|
|
|
2018-05-12 23:26:31 +00:00
|
|
|
if (FILE *t = pid_popen(std::get<0>(tuple).c_str(), "r", &childpid)) {
|
2018-05-12 16:03:00 +00:00
|
|
|
fp.reset(t, fclose);
|
2018-05-12 23:26:31 +00:00
|
|
|
} else {
|
2018-05-12 16:03:00 +00:00
|
|
|
return;
|
2018-05-12 23:26:31 +00:00
|
|
|
}
|
2018-05-12 16:03:00 +00:00
|
|
|
|
2018-05-12 23:26:31 +00:00
|
|
|
while ((feof(fp.get()) == 0) && (ferror(fp.get()) == 0)) {
|
2018-05-12 16:03:00 +00:00
|
|
|
int length = fread(b, 1, sizeof b, fp.get());
|
|
|
|
buf.append(b, length);
|
|
|
|
}
|
|
|
|
|
2018-05-25 00:24:09 +00:00
|
|
|
if (*buf.rbegin() == '\n') { buf.resize(buf.size() - 1); }
|
2018-05-12 16:03:00 +00:00
|
|
|
|
|
|
|
std::lock_guard<std::mutex> l(result_mutex);
|
|
|
|
result = buf;
|
2011-02-26 17:54:48 +00:00
|
|
|
}
|
2016-01-09 17:19:48 +00:00
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
// remove backspaced chars, example: "dog^H^H^Hcat" becomes "cat"
|
|
|
|
// string has to end with \0 and it's length should fit in a int
|
2009-10-15 19:51:21 +00:00
|
|
|
#define BACKSPACE 8
|
2018-05-12 16:03:00 +00:00
|
|
|
static void remove_deleted_chars(char *string) {
|
|
|
|
int i = 0;
|
|
|
|
while (string[i] != 0) {
|
|
|
|
if (string[i] == BACKSPACE) {
|
|
|
|
if (i != 0) {
|
|
|
|
strcpy(&(string[i - 1]), &(string[i + 1]));
|
|
|
|
i--;
|
2018-05-12 23:26:31 +00:00
|
|
|
} else {
|
2018-05-12 16:03:00 +00:00
|
|
|
strcpy(
|
|
|
|
&(string[i]),
|
|
|
|
&(string[i + 1])); // necessary for ^H's at the start of a string
|
2018-05-12 23:26:31 +00:00
|
|
|
}
|
|
|
|
} else {
|
2018-05-12 16:03:00 +00:00
|
|
|
i++;
|
2018-05-12 23:26:31 +00:00
|
|
|
}
|
2018-05-12 16:03:00 +00:00
|
|
|
}
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Parses command output to find a number between 0.0 and 100.0.
|
|
|
|
* Used by ${exec[i]{bar,gauge,graph}}.
|
|
|
|
*
|
|
|
|
* @param[in] buf output of a command executed by an exec_cb object
|
|
|
|
* @return number between 0.0 and 100.0
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
static inline double get_barnum(const char *buf) {
|
|
|
|
double barnum;
|
|
|
|
|
|
|
|
if (sscanf(buf, "%lf", &barnum) != 1) {
|
2019-03-03 16:18:54 +00:00
|
|
|
NORM_ERR(
|
|
|
|
"reading exec value failed (perhaps it's not the "
|
|
|
|
"correct format?)");
|
|
|
|
return 0.0;
|
2018-05-12 16:03:00 +00:00
|
|
|
}
|
|
|
|
if (barnum > 100.0 || barnum < 0.0) {
|
|
|
|
NORM_ERR(
|
|
|
|
"your exec value is not between 0 and 100, "
|
|
|
|
"therefore it will be ignored");
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return barnum;
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Store command output in p. For execp objects, we process the output
|
|
|
|
* in case it contains special commands like ${color}
|
|
|
|
*
|
|
|
|
* @param[in] buffer the output of a command
|
|
|
|
* @param[in] obj text_object that specifies whether or not to parse
|
|
|
|
* @param[out] p the string in which we store command output
|
|
|
|
* @param[in] p_max_size the maximum size of p...
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void fill_p(const char *buffer, struct text_object *obj, char *p,
|
2018-08-07 23:22:23 +00:00
|
|
|
unsigned int p_max_size) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if (obj->parse) {
|
2018-05-12 16:03:00 +00:00
|
|
|
evaluate(buffer, p, p_max_size);
|
|
|
|
} else {
|
|
|
|
snprintf(p, p_max_size, "%s", buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
remove_deleted_chars(p);
|
2009-12-04 01:09:41 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Parses arg to find the command to be run, as well as special options
|
|
|
|
* like height, width, color, and update interval
|
|
|
|
*
|
2018-05-12 16:03:00 +00:00
|
|
|
* @param[out] obj stores the command and an execi_data structure (if
|
|
|
|
* applicable)
|
2016-01-09 17:19:48 +00:00
|
|
|
* @param[in] arg the argument to an ${exec*} object
|
2018-05-12 16:03:00 +00:00
|
|
|
* @param[in] execflag bitwise flag used to specify the exec variant we need to
|
|
|
|
* process
|
2016-01-09 17:19:48 +00:00
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void scan_exec_arg(struct text_object *obj, const char *arg,
|
|
|
|
unsigned int execflag) {
|
|
|
|
const char *cmd = arg;
|
2018-12-05 21:21:04 +00:00
|
|
|
char *orig_cmd = nullptr;
|
2018-05-12 16:03:00 +00:00
|
|
|
struct execi_data *ed;
|
|
|
|
|
|
|
|
/* in case we have an execi object, we need to parse out the interval */
|
2018-05-12 23:26:31 +00:00
|
|
|
if ((execflag & EF_EXECI) != 0u) {
|
2018-05-12 16:03:00 +00:00
|
|
|
ed = new execi_data;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
/* store the interval in ed->interval */
|
|
|
|
if (sscanf(arg, "%f %n", &ed->interval, &n) <= 0) {
|
|
|
|
NORM_ERR("missing execi interval: ${execi* <interval> command}");
|
|
|
|
delete ed;
|
2018-05-12 23:26:31 +00:00
|
|
|
ed = nullptr;
|
2018-05-12 16:03:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set cmd to everything after the interval */
|
|
|
|
cmd = strndup(arg + n, text_buffer_size.get(*state));
|
2019-02-23 19:43:44 +00:00
|
|
|
orig_cmd = const_cast<char *>(cmd);
|
2018-05-12 16:03:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* parse any special options for the graphical exec types */
|
2018-05-12 23:26:31 +00:00
|
|
|
if ((execflag & EF_BAR) != 0u) {
|
2018-05-12 16:03:00 +00:00
|
|
|
cmd = scan_bar(obj, cmd, 100);
|
2016-01-09 17:19:48 +00:00
|
|
|
#ifdef BUILD_X11
|
2018-05-12 23:26:31 +00:00
|
|
|
} else if ((execflag & EF_GAUGE) != 0u) {
|
2018-05-12 16:03:00 +00:00
|
|
|
cmd = scan_gauge(obj, cmd, 100);
|
2018-05-12 23:26:31 +00:00
|
|
|
} else if ((execflag & EF_GRAPH) != 0u) {
|
2018-05-12 16:03:00 +00:00
|
|
|
cmd = scan_graph(obj, cmd, 100);
|
2018-05-12 23:26:31 +00:00
|
|
|
if (cmd == nullptr) {
|
2018-05-12 16:03:00 +00:00
|
|
|
NORM_ERR("error parsing arguments to execgraph object");
|
|
|
|
}
|
2010-01-07 02:38:12 +00:00
|
|
|
#endif /* BUILD_X11 */
|
2018-05-12 16:03:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* finally, store the resulting command, or an empty string if something went
|
|
|
|
* wrong */
|
2018-05-12 23:26:31 +00:00
|
|
|
if ((execflag & EF_EXEC) != 0u) {
|
|
|
|
obj->data.s =
|
|
|
|
strndup(cmd != nullptr ? cmd : "", text_buffer_size.get(*state));
|
|
|
|
} else if ((execflag & EF_EXECI) != 0u) {
|
|
|
|
ed->cmd = strndup(cmd != nullptr ? cmd : "", text_buffer_size.get(*state));
|
2018-05-12 16:03:00 +00:00
|
|
|
obj->data.opaque = ed;
|
|
|
|
}
|
2018-12-05 21:21:04 +00:00
|
|
|
free_and_zero(orig_cmd);
|
2010-02-10 18:45:42 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Register an exec_cb object using the command that we have parsed
|
|
|
|
*
|
|
|
|
* @param[out] obj stores the callback handle
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void register_exec(struct text_object *obj) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if ((obj->data.s != nullptr) && (obj->data.s[0] != 0)) {
|
2018-05-12 16:03:00 +00:00
|
|
|
obj->exec_handle = new conky::callback_handle<exec_cb>(
|
|
|
|
conky::register_cb<exec_cb>(1, true, obj->data.s));
|
|
|
|
} else {
|
|
|
|
DBGP("unable to register exec callback");
|
|
|
|
}
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Register an exec_cb object using the command that we have parsed.
|
|
|
|
*
|
|
|
|
* This version takes care of execi intervals. Note that we depend on
|
|
|
|
* obj->thread, so be sure to run this function *after* setting obj->thread.
|
|
|
|
*
|
|
|
|
* @param[out] obj stores the callback handle
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void register_execi(struct text_object *obj) {
|
2018-05-12 23:26:31 +00:00
|
|
|
auto *ed = static_cast<struct execi_data *>(obj->data.opaque);
|
2018-05-12 16:03:00 +00:00
|
|
|
|
2018-05-12 23:26:31 +00:00
|
|
|
if ((ed != nullptr) && (ed->cmd != nullptr) && (ed->cmd[0] != 0)) {
|
2018-05-12 16:03:00 +00:00
|
|
|
uint32_t period =
|
|
|
|
std::max(lround(ed->interval / active_update_interval()), 1l);
|
|
|
|
obj->exec_handle = new conky::callback_handle<exec_cb>(
|
|
|
|
conky::register_cb<exec_cb>(period, !obj->thread, ed->cmd));
|
|
|
|
} else {
|
|
|
|
DBGP("unable to register execi callback");
|
|
|
|
}
|
2010-02-07 09:29:43 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Get the results of an exec_cb object (command output)
|
|
|
|
*
|
|
|
|
* @param[in] obj holds an exec_handle, assuming one was registered
|
|
|
|
* @param[out] p the string in which we store command output
|
|
|
|
* @param[in] p_max_size the maximum size of p...
|
|
|
|
*/
|
2018-08-07 23:22:23 +00:00
|
|
|
void print_exec(struct text_object *obj, char *p, unsigned int p_max_size) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if (obj->exec_handle != nullptr) {
|
2018-05-12 16:03:00 +00:00
|
|
|
fill_p((*obj->exec_handle)->get_result_copy().c_str(), obj, p, p_max_size);
|
|
|
|
}
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Get the results of a graphical (bar, gauge, graph) exec_cb object
|
|
|
|
*
|
|
|
|
* @param[in] obj hold an exec_handle, assuming one was registered
|
|
|
|
* @return a value between 0.0 and 100.0
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
double execbarval(struct text_object *obj) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if (obj->exec_handle != nullptr) {
|
2018-05-12 16:03:00 +00:00
|
|
|
return get_barnum((*obj->exec_handle)->get_result_copy().c_str());
|
|
|
|
}
|
2018-05-25 00:24:09 +00:00
|
|
|
return 0.0;
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Free up any dynamically allocated data
|
|
|
|
*
|
|
|
|
* @param[in] obj holds the data that we need to free up
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void free_exec(struct text_object *obj) {
|
|
|
|
free_and_zero(obj->data.s);
|
|
|
|
delete obj->exec_handle;
|
2018-05-12 23:26:31 +00:00
|
|
|
obj->exec_handle = nullptr;
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-09 17:19:48 +00:00
|
|
|
/**
|
|
|
|
* Free up any dynamically allocated data, specifically for execi objects
|
|
|
|
*
|
|
|
|
* @param[in] obj holds the data that we need to free up
|
|
|
|
*/
|
2018-05-12 16:03:00 +00:00
|
|
|
void free_execi(struct text_object *obj) {
|
2018-05-12 23:26:31 +00:00
|
|
|
auto *ed = static_cast<struct execi_data *>(obj->data.opaque);
|
2009-10-06 22:09:14 +00:00
|
|
|
|
2018-05-12 23:26:31 +00:00
|
|
|
/* if ed is nullptr, there is nothing to do */
|
2018-05-25 00:24:09 +00:00
|
|
|
if (ed == nullptr) { return; }
|
2009-10-06 22:09:14 +00:00
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
delete obj->exec_handle;
|
2018-05-12 23:26:31 +00:00
|
|
|
obj->exec_handle = nullptr;
|
2016-01-09 17:19:48 +00:00
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
free_and_zero(ed->cmd);
|
|
|
|
delete ed;
|
2018-05-12 23:26:31 +00:00
|
|
|
ed = nullptr;
|
|
|
|
obj->data.opaque = nullptr;
|
2009-10-15 19:51:21 +00:00
|
|
|
}
|