diff options
-rw-r--r-- | arg.h | 65 | ||||
-rw-r--r-- | slock.c | 85 |
2 files changed, 119 insertions, 31 deletions
@@ -0,0 +1,65 @@ | |||
1 | /* | ||
2 | * Copy me if you can. | ||
3 | * by 20h | ||
4 | */ | ||
5 | |||
6 | #ifndef ARG_H__ | ||
7 | #define ARG_H__ | ||
8 | |||
9 | extern char *argv0; | ||
10 | |||
11 | /* use main(int argc, char *argv[]) */ | ||
12 | #define ARGBEGIN for (argv0 = *argv, argv++, argc--;\ | ||
13 | argv[0] && argv[0][0] == '-'\ | ||
14 | && argv[0][1];\ | ||
15 | argc--, argv++) {\ | ||
16 | char argc_;\ | ||
17 | char **argv_;\ | ||
18 | int brk_;\ | ||
19 | if (argv[0][1] == '-' && argv[0][2] == '\0') {\ | ||
20 | argv++;\ | ||
21 | argc--;\ | ||
22 | break;\ | ||
23 | }\ | ||
24 | for (brk_ = 0, argv[0]++, argv_ = argv;\ | ||
25 | argv[0][0] && !brk_;\ | ||
26 | argv[0]++) {\ | ||
27 | if (argv_ != argv)\ | ||
28 | break;\ | ||
29 | argc_ = argv[0][0];\ | ||
30 | switch (argc_) | ||
31 | |||
32 | /* Handles obsolete -NUM syntax */ | ||
33 | #define ARGNUM case '0':\ | ||
34 | case '1':\ | ||
35 | case '2':\ | ||
36 | case '3':\ | ||
37 | case '4':\ | ||
38 | case '5':\ | ||
39 | case '6':\ | ||
40 | case '7':\ | ||
41 | case '8':\ | ||
42 | case '9' | ||
43 | |||
44 | #define ARGEND }\ | ||
45 | } | ||
46 | |||
47 | #define ARGC() argc_ | ||
48 | |||
49 | #define ARGNUMF() (brk_ = 1, estrtonum(argv[0], 0, INT_MAX)) | ||
50 | |||
51 | #define EARGF(x) ((argv[0][1] == '\0' && argv[1] == NULL)?\ | ||
52 | ((x), abort(), (char *)0) :\ | ||
53 | (brk_ = 1, (argv[0][1] != '\0')?\ | ||
54 | (&argv[0][1]) :\ | ||
55 | (argc--, argv++, argv[0]))) | ||
56 | |||
57 | #define ARGF() ((argv[0][1] == '\0' && argv[1] == NULL)?\ | ||
58 | (char *)0 :\ | ||
59 | (brk_ = 1, (argv[0][1] != '\0')?\ | ||
60 | (&argv[0][1]) :\ | ||
61 | (argc--, argv++, argv[0]))) | ||
62 | |||
63 | #define LNGARG() &argv[0][0] | ||
64 | |||
65 | #endif | ||
@@ -23,8 +23,11 @@ | |||
23 | #include <bsd_auth.h> | 23 | #include <bsd_auth.h> |
24 | #endif | 24 | #endif |
25 | 25 | ||
26 | #include "arg.h" | ||
26 | #include "util.h" | 27 | #include "util.h" |
27 | 28 | ||
29 | char *argv0; | ||
30 | |||
28 | enum { | 31 | enum { |
29 | INIT, | 32 | INIT, |
30 | INPUT, | 33 | INPUT, |
@@ -54,7 +57,6 @@ die(const char *errstr, ...) | |||
54 | { | 57 | { |
55 | va_list ap; | 58 | va_list ap; |
56 | 59 | ||
57 | fputs("slock: ", stderr); | ||
58 | va_start(ap, errstr); | 60 | va_start(ap, errstr); |
59 | vfprintf(stderr, errstr, ap); | 61 | vfprintf(stderr, errstr, ap); |
60 | va_end(ap); | 62 | va_end(ap); |
@@ -280,8 +282,7 @@ lockscreen(Display *dpy, int screen) | |||
280 | static void | 282 | static void |
281 | usage(void) | 283 | usage(void) |
282 | { | 284 | { |
283 | fprintf(stderr, "usage: slock [-v|POST_LOCK_CMD]\n"); | 285 | die("usage: slock [-v | cmd [arg ...]]\n"); |
284 | exit(1); | ||
285 | } | 286 | } |
286 | 287 | ||
287 | int | 288 | int |
@@ -290,64 +291,86 @@ main(int argc, char **argv) { | |||
290 | const char *pws; | 291 | const char *pws; |
291 | #endif | 292 | #endif |
292 | Display *dpy; | 293 | Display *dpy; |
293 | int screen; | 294 | int s, nlocks; |
294 | 295 | ||
295 | if ((argc >= 2) && !strcmp("-v", argv[1])) | 296 | ARGBEGIN { |
296 | die("version %s, © 2006-2016 slock engineers\n", VERSION); | 297 | case 'v': |
297 | 298 | fprintf(stderr, "slock-"VERSION"\n"); | |
298 | /* treat first argument starting with a '-' as option */ | 299 | return 0; |
299 | if ((argc >= 2) && argv[1][0] == '-') | 300 | default: |
300 | usage(); | 301 | usage(); |
302 | } ARGEND | ||
301 | 303 | ||
302 | #ifdef __linux__ | 304 | #ifdef __linux__ |
303 | dontkillme(); | 305 | dontkillme(); |
304 | #endif | 306 | #endif |
305 | 307 | ||
306 | if (!getpwuid(getuid())) | 308 | /* Check if the current user has a password entry */ |
307 | die("no passwd entry for you\n"); | 309 | errno = 0; |
310 | if (!getpwuid(getuid())) { | ||
311 | if (errno == 0) | ||
312 | die("slock: no password entry for current user\n"); | ||
313 | else | ||
314 | die("slock: getpwuid: %s\n", strerror(errno)); | ||
315 | } | ||
308 | 316 | ||
309 | #ifndef HAVE_BSD_AUTH | 317 | #ifndef HAVE_BSD_AUTH |
310 | pws = getpw(); | 318 | pws = getpw(); |
311 | #endif | 319 | #endif |
312 | 320 | ||
313 | if (!(dpy = XOpenDisplay(0))) | 321 | if (!(dpy = XOpenDisplay(NULL))) |
314 | die("cannot open display\n"); | 322 | die("slock: cannot open display\n"); |
323 | |||
324 | /* check for Xrandr support */ | ||
315 | rr = XRRQueryExtension(dpy, &rrevbase, &rrerrbase); | 325 | rr = XRRQueryExtension(dpy, &rrevbase, &rrerrbase); |
316 | /* Get the number of screens in display "dpy" and blank them all. */ | 326 | |
327 | /* get number of screens in display "dpy" and blank them */ | ||
317 | nscreens = ScreenCount(dpy); | 328 | nscreens = ScreenCount(dpy); |
318 | if (!(locks = malloc(sizeof(Lock*) * nscreens))) | 329 | if (!(locks = malloc(sizeof(Lock *) * nscreens))) { |
319 | die("Out of memory.\n"); | 330 | XCloseDisplay(dpy); |
320 | int nlocks = 0; | 331 | die("slock: out of memory\n"); |
321 | for (screen = 0; screen < nscreens; screen++) { | 332 | } |
322 | if ((locks[screen] = lockscreen(dpy, screen)) != NULL) | 333 | for (nlocks = 0, s = 0; s < nscreens; s++) { |
334 | if ((locks[s] = lockscreen(dpy, s)) != NULL) | ||
323 | nlocks++; | 335 | nlocks++; |
324 | } | 336 | } |
325 | XSync(dpy, False); | 337 | XSync(dpy, 0); |
326 | 338 | ||
327 | /* Did we actually manage to lock something? */ | 339 | /* did we actually manage to lock anything? */ |
328 | if (nlocks == 0) { /* nothing to protect */ | 340 | if (nlocks == 0) { |
341 | /* nothing to protect */ | ||
329 | free(locks); | 342 | free(locks); |
330 | XCloseDisplay(dpy); | 343 | XCloseDisplay(dpy); |
331 | return 1; | 344 | return 1; |
332 | } | 345 | } |
333 | 346 | ||
334 | if (argc >= 2 && fork() == 0) { | 347 | /* run post-lock command */ |
335 | if (dpy) | 348 | if (argc > 0) { |
336 | close(ConnectionNumber(dpy)); | 349 | switch (fork()) { |
337 | execvp(argv[1], argv+1); | 350 | case -1: |
338 | die("execvp %s failed: %s\n", argv[1], strerror(errno)); | 351 | free(locks); |
352 | XCloseDisplay(dpy); | ||
353 | die("slock: fork failed: %s\n", strerror(errno)); | ||
354 | case 0: | ||
355 | if (close(ConnectionNumber(dpy)) < 0) | ||
356 | die("slock: close: %s\n", strerror(errno)); | ||
357 | execvp(argv[0], argv); | ||
358 | fprintf(stderr, "slock: execvp %s: %s\n", argv[0], | ||
359 | strerror(errno)); | ||
360 | _exit(1); | ||
361 | } | ||
339 | } | 362 | } |
340 | 363 | ||
341 | /* Everything is now blank. Now wait for the correct password. */ | 364 | /* everything is now blank. Wait for the correct password */ |
342 | #ifdef HAVE_BSD_AUTH | 365 | #ifdef HAVE_BSD_AUTH |
343 | readpw(dpy); | 366 | readpw(dpy); |
344 | #else | 367 | #else |
345 | readpw(dpy, pws); | 368 | readpw(dpy, pws); |
346 | #endif | 369 | #endif |
347 | 370 | ||
348 | /* Password ok, unlock everything and quit. */ | 371 | /* password ok, unlock everything and quit */ |
349 | for (screen = 0; screen < nscreens; screen++) | 372 | for (s = 0; s < nscreens; s++) |
350 | unlockscreen(dpy, locks[screen]); | 373 | unlockscreen(dpy, locks[s]); |
351 | 374 | ||
352 | free(locks); | 375 | free(locks); |
353 | XCloseDisplay(dpy); | 376 | XCloseDisplay(dpy); |