Skip to content

Commit d8494b6

Browse files
committed
shutdown-wrapper is a wrapper which tests the shutdown-hooks
1 parent 684ec3d commit d8494b6

17 files changed

+4198
-99
lines changed

HISTORY

+5
Original file line numberDiff line numberDiff line change
@@ -103,3 +103,8 @@ VDR Plugin 'dbus2vdr' Revision History
103103
2012-02-08: Version 0.0.3h
104104

105105
- add "AskUser" to remote interface
106+
107+
2012-02-19: Version 0.0.3i
108+
109+
- shutdown-wrapper is a wrapper which tests the shutdown-hooks
110+
set its SUID bit if needed

Makefile

+8-3
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ DEFINES += -D_GNU_SOURCE -DPLUGIN_NAME_I18N='"$(PLUGIN)"'
5656
### The object files (add further files here):
5757

5858
OBJS = $(PLUGIN).o epg.o helper.o message.o monitor.o plugin.o recording.o remote.o setup.o shutdown.o skin.o timer.o
59+
SWOBJS = libvdr-exitpipe.o libvdr-i18n.o libvdr-thread.o libvdr-tools.o shutdown-wrapper.o
5960

6061
### The main target:
6162

@@ -87,7 +88,7 @@ I18Npot = $(PODIR)/$(PLUGIN).pot
8788
%.mo: %.po
8889
msgfmt -c -o $@ $<
8990

90-
$(I18Npot): $(wildcard *.c)
91+
$(I18Npot): $(patsubst %.o,%.c,$(OBJS))
9192
xgettext -C -cTRANSLATORS --no-wrap --no-location -k -ktr -ktrNOOP --package-name=vdr-$(PLUGIN) --package-version=$(VERSION) --msgid-bugs-address='<see README>' -o $@ $^
9293

9394
%.po: $(I18Npot)
@@ -104,7 +105,10 @@ i18n: $(I18Nmsgs) $(I18Npot)
104105

105106
### Targets:
106107

107-
libvdr-$(PLUGIN).so: $(OBJS)
108+
shutdown-wrapper: $(SWOBJS)
109+
$(CXX) $(CXXFLAGS) $(LDFLAGS) $(SWOBJS) -ljpeg -lrt -pthread -o $@
110+
111+
libvdr-$(PLUGIN).so: $(OBJS) shutdown-wrapper
108112
$(CXX) $(CXXFLAGS) $(LDFLAGS) -shared $(OBJS) $(LDADD) -o $@
109113
@cp --remove-destination $@ $(LIBDIR)/$@.$(APIVERSION)
110114

@@ -117,4 +121,5 @@ dist: $(I18Npo) clean
117121
@echo Distribution package created as $(PACKAGE).tgz
118122

119123
clean:
120-
@-rm -f $(OBJS) $(DEPFILE) *.so *.tgz core* *~ $(PODIR)/*.mo $(PODIR)/*.pot
124+
@-rm -f $(OBJS) $(SWOBJS) shutdown-wrapper $(DEPFILE) *.so *.tgz core* *~ $(PODIR)/*.mo $(PODIR)/*.pot
125+

README

+4
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@ Commandline Arguments
4545
-s, --shutdown-hooks=/path/to/shutdown-hooks
4646
uses all files in the given directory as shutdown-hooks
4747
usually it's /usr/share/vdr/shutdown-hooks
48+
-w, --shutdown-hooks-wrapper=/path/to/shutdown-hooks-wrapper
49+
path to a program that will call the shutdown-hooks with suid
4850

4951
Implemented features
5052
--------------------
@@ -121,6 +123,8 @@ If the SVDRP command doesn't take a parameter you can ommit it or pass an empty
121123

122124
You may shutdown vdr on reply codes 250 and 990.
123125
If SHUTDOWNCMD is given use this command for shutdown.
126+
'shutdown-wrapper' is a tool which calls the shutdown-hooks.
127+
If the hooks need elevated privileges set the SUID bit of the wrapper accordingly.
124128

125129
- list all supported parameters from setup.conf
126130
vdr-dbus-send.sh /Setup setup.List

dbus2vdr.c

+14-3
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121
#include <vdr/osdbase.h>
2222
#include <vdr/plugin.h>
2323

24-
static const char *VERSION = "0.0.3h";
24+
static const char *VERSION = "0.0.3i";
2525
static const char *DESCRIPTION = trNOOP("control vdr via D-Bus");
2626
static const char *MAINMENUENTRY = NULL;
2727

@@ -67,20 +67,23 @@ const char *cPluginDbus2vdr::CommandLineHelp(void)
6767
{
6868
return " --shutdown-hooks=/path/to/dir/with/shutdown-hooks\n"
6969
" directory with shutdown-hooks to be called by ConfirmShutdown\n"
70-
" usually it's /usr/share/vdr/shutdown-hooks";
70+
" usually it's /usr/share/vdr/shutdown-hooks\n"
71+
" --shutdown-hooks-wrapper=/path/to/shutdown-hooks-wrapper\n"
72+
" path to a program that will call the shutdown-hooks with suid";
7173
}
7274

7375
bool cPluginDbus2vdr::ProcessArgs(int argc, char *argv[])
7476
{
7577
static struct option options[] =
7678
{
7779
{"shutdown-hooks", required_argument, 0, 's'},
80+
{"shutdown-hooks-wrapper", required_argument, 0, 'w'},
7881
{0, 0, 0, 0}
7982
};
8083

8184
while (true) {
8285
int option_index = 0;
83-
int c = getopt_long(argc, argv, "s:", options, &option_index);
86+
int c = getopt_long(argc, argv, "s:w:", options, &option_index);
8487
if (c == -1)
8588
break;
8689
switch (c) {
@@ -92,6 +95,14 @@ bool cPluginDbus2vdr::ProcessArgs(int argc, char *argv[])
9295
}
9396
break;
9497
}
98+
case 'w':
99+
{
100+
if (optarg != NULL) {
101+
isyslog("dbus2vdr: use shutdown-hooks-wrapper %s", optarg);
102+
cDBusMessageShutdown::SetShutdownHooksWrapper(optarg);
103+
}
104+
break;
105+
}
95106
}
96107
}
97108
return true;

libvdr-exitpipe.c

+114
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,114 @@
1+
#include "libvdr-exitpipe.h"
2+
3+
#include "libvdr-thread.h"
4+
5+
#include <sys/wait.h>
6+
#include <unistd.h>
7+
8+
9+
cExitPipe::cExitPipe(void)
10+
{
11+
pid = -1;
12+
f = NULL;
13+
}
14+
15+
cExitPipe::~cExitPipe()
16+
{
17+
Close();
18+
}
19+
20+
bool cExitPipe::Open(const char *Command, const char *Mode)
21+
{
22+
int fd[2];
23+
24+
if (pipe(fd) < 0) {
25+
LOG_ERROR;
26+
return false;
27+
}
28+
if ((pid = fork()) < 0) { // fork failed
29+
LOG_ERROR;
30+
close(fd[0]);
31+
close(fd[1]);
32+
return false;
33+
}
34+
35+
const char *mode = "w";
36+
int iopipe = 0;
37+
38+
if (pid > 0) { // parent process
39+
if (strcmp(Mode, "r") == 0) {
40+
mode = "r";
41+
iopipe = 1;
42+
}
43+
close(fd[iopipe]);
44+
if ((f = fdopen(fd[1 - iopipe], mode)) == NULL) {
45+
LOG_ERROR;
46+
close(fd[1 - iopipe]);
47+
}
48+
return f != NULL;
49+
}
50+
else { // child process
51+
int iofd = STDOUT_FILENO;
52+
if (strcmp(Mode, "w") == 0) {
53+
mode = "r";
54+
iopipe = 1;
55+
iofd = STDIN_FILENO;
56+
}
57+
close(fd[iopipe]);
58+
if (dup2(fd[1 - iopipe], iofd) == -1) { // now redirect
59+
LOG_ERROR;
60+
close(fd[1 - iopipe]);
61+
_exit(-1);
62+
}
63+
else {
64+
int MaxPossibleFileDescriptors = getdtablesize();
65+
for (int i = STDERR_FILENO + 1; i < MaxPossibleFileDescriptors; i++)
66+
close(i); //close all dup'ed filedescriptors
67+
if (execl("/bin/sh", "sh", "-c", Command, NULL) == -1) {
68+
LOG_ERROR_STR(Command);
69+
close(fd[1 - iopipe]);
70+
_exit(-1);
71+
}
72+
}
73+
_exit(0);
74+
}
75+
}
76+
77+
int cExitPipe::Close(void)
78+
{
79+
int ret = -1;
80+
81+
if (f) {
82+
fclose(f);
83+
f = NULL;
84+
}
85+
86+
if (pid > 0) {
87+
int status = 0;
88+
int i = 5;
89+
while (i > 0) {
90+
ret = waitpid(pid, &status, WNOHANG);
91+
if (ret < 0) {
92+
if (errno != EINTR && errno != ECHILD) {
93+
LOG_ERROR;
94+
break;
95+
}
96+
}
97+
else if (ret == pid)
98+
break;
99+
i--;
100+
cCondWait::SleepMs(100);
101+
}
102+
if (!i) {
103+
kill(pid, SIGKILL);
104+
ret = -1;
105+
}
106+
else if (ret == -1 || !WIFEXITED(status))
107+
ret = -1;
108+
else if (WIFEXITED(status))
109+
ret = WEXITSTATUS(status);
110+
pid = -1;
111+
}
112+
113+
return ret;
114+
}

libvdr-exitpipe.h

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
#ifndef __DBUS2VDR_EXITPIPE_H
2+
#define __DBUS2VDR_EXITPIPE_H
3+
4+
#include "libvdr-tools.h"
5+
6+
// copy of vdr's cPipe but returns exit code of child on Close
7+
class cExitPipe
8+
{
9+
private:
10+
pid_t pid;
11+
FILE *f;
12+
public:
13+
cExitPipe(void);
14+
~cExitPipe();
15+
operator FILE* () { return f; }
16+
bool Open(const char *Command, const char *Mode);
17+
int Close(void);
18+
};
19+
#endif

0 commit comments

Comments
 (0)