source: neercs/trunk/src/main.c @ 3974

Last change on this file since 3974 was 3974, checked in by Jean-Yves Lamoureux, 11 years ago
  • Moved client initialisation to client.c
  • Property svn:keywords set to Id
File size: 9.0 KB
Line 
1/*
2 *  neercs        console-based window manager
3 *  Copyright (c) 2006 Sam Hocevar <sam@zoy.org>
4 *                2008 Jean-Yves Lamoureux <jylam@lnxscene.org>
5 *                2008 Pascal Terjan <pterjan@linuxfr.org>
6 *                All Rights Reserved
7 *
8 *  $Id: main.c 3974 2009-11-20 12:27:05Z jylam $
9 *
10 *  This program is free software. It comes without any warranty, to
11 *  the extent permitted by applicable law. You can redistribute it
12 *  and/or modify it under the terms of the Do What The Fuck You Want
13 *  To Public License, Version 2, as published by Sam Hocevar. See
14 *  http://sam.zoy.org/wtfpl/COPYING for more details.
15 */
16
17#include "config.h"
18
19#include <stdio.h>
20#include <string.h>
21#include <stdlib.h>
22#include <unistd.h>
23#include <fcntl.h>
24#include <signal.h>
25#include <sys/ioctl.h>
26#include <sys/types.h>
27#include <sys/wait.h>
28#include <sys/time.h>
29#include <time.h>
30
31
32
33#if !defined HAVE_GETOPT_LONG
34#   include "mygetopt.h"
35#elif defined HAVE_GETOPT_H
36#   include <getopt.h>
37#endif
38#if defined HAVE_GETOPT_LONG
39#   define mygetopt getopt_long
40#   define myoptind optind
41#   define myoptarg optarg
42#   define myoption option
43#endif
44#include <errno.h>
45#include <caca.h>
46#include <caca.h>
47
48#include "neercs.h"
49
50
51void version(void)
52{
53    printf("%s\n", PACKAGE_STRING);
54    printf("Copyright (C) 2006, 2008 Sam Hocevar <sam@zoy.org>\n");
55    printf
56        ("                         Jean-Yves Lamoureux <jylam@lnxscene.org>\n\n");
57    printf
58        ("This is free software.  You may redistribute copies of it under the\n");
59    printf
60        ("terms of the Do What The Fuck You Want To Public License, Version 2\n");
61    printf("<http://sam.zoy.org/wtfpl/>.\n");
62    printf("There is NO WARRANTY, to the extent permitted by law.\n");
63    printf("\n");
64    printf
65        ("For more informations, visit http://libcaca.zoy.org/wiki/neercs\n");
66}
67
68void usage(int argc, char **argv)
69{
70    printf("%s\n", PACKAGE_STRING);
71    printf("Usage : %s [command1] [command2] ... [commandN]\n", argv[0]);
72    printf("Example : %s zsh top \n\n", argv[0]);
73    printf("Options :\n");
74    printf("\t--config\t-c <file>\t\tuse given config file\n");
75    printf("\t--pid\t\t-P [pid]\t\tgrab process\n");
76    printf("\t\t\t-r [session]\t\treattach to a detached neercs\n");
77    printf
78        ("\t\t\t-R [session]\t\treattach if possible, otherwise start a new session\n");
79    printf("\t\t\t-S <name>\t\tname this session <name> instead of <pid>\n");
80    printf("\t--lock-after\t-l [n]\t\t\tlock screen after n seconds\n");
81    printf("\t--version\t-v \t\t\tdisplay version and exit\n");
82    printf("\t--help\t\t-h \t\t\tthis help\n");
83}
84
85int main(int argc, char **argv)
86{
87    struct screen_list *screen_list = NULL;
88    int args;
89
90    int mainret = -1;
91
92    screen_list = create_screen_list();
93    screen_list->default_shell = getenv("SHELL");
94
95    args = argc - 1;
96    if (screen_list->default_shell == NULL && args <= 0)
97    {
98        fprintf(stderr,
99                "Environment variable SHELL not set and no arguments given. kthxbye.\n");
100        goto end;
101    }
102
103    if (handle_command_line(argc, argv, screen_list) < 0)
104        goto end;
105
106    /* Read global configuration first */
107    read_configuration_file("/etc/neercsrc", screen_list);
108
109    /* Then local one */
110    if (screen_list->user_path)
111    {
112        read_configuration_file(screen_list->user_path, screen_list);
113        free(screen_list->user_path);
114    }
115
116    if (screen_list->attach)
117    {
118        if (screen_list->nb_to_grab || screen_list->to_start)
119        {
120            fprintf(stderr,
121                    "-R can not be associated with commands or pids!\n");
122            goto end;
123        }
124
125        attach(screen_list);
126
127        if (screen_list->forceattach && !screen_list->attach)
128            goto end;
129    }
130
131    /* Build default session name */
132    if (!screen_list->session_name)
133    {
134        char mypid[32];         /* FIXME Compute the length of PID_MAX ? */
135        snprintf(mypid, 31, "%d", getpid());
136        mypid[31] = '\0';
137        screen_list->session_name = strdup(mypid);
138        if (!screen_list->session_name)
139        {
140            fprintf(stderr, "Can't allocate memory at %s:%d\n", __FUNCTION__,
141                    __LINE__);
142            goto end;
143        }
144    }
145    if (!screen_list->socket_path[SOCK_CLIENT])
146        screen_list->socket_path[SOCK_CLIENT] =
147            build_socket_path(screen_list->socket_dir,
148                              screen_list->session_name, SOCK_CLIENT);
149
150    if (!screen_list->socket_path[SOCK_SERVER])
151        screen_list->socket_path[SOCK_SERVER] =
152            build_socket_path(screen_list->socket_dir,
153                              screen_list->session_name, SOCK_SERVER);
154
155    /* Fork the server if needed */
156    if (!screen_list->attach)
157    {
158        debug("Spawning a new server");
159        if (start_server(screen_list))
160            goto end;
161        if (start_client(screen_list))
162            goto end;
163    }
164
165
166    mainloop(screen_list);
167
168    /* Clean up */
169    mainret = 0;
170  end:
171
172    if (screen_list)
173    {
174        free_screen_list(screen_list);
175    }
176
177    return mainret;
178}
179
180
181
182int handle_command_line(int argc, char *argv[],
183                        struct screen_list *screen_list)
184{
185    int s = 0, i;
186    for (;;)
187    {
188        int option_index = 0;
189        int pidopt;
190        static struct myoption long_options[] = {
191            {"config", 1, NULL, 'c'},
192#if defined USE_GRAB
193            {"pid", 0, NULL, 'P'},
194#endif
195            {"lock-after", 1, NULL, 'l'},
196            {"help", 0, NULL, 'h'},
197            {"version", 0, NULL, 'v'},
198            {NULL, 0, NULL, 0},
199        };
200#if defined USE_GRAB
201        int c =
202            mygetopt(argc, argv, "c:S:R::l::r::P::hv", long_options,
203                     &option_index);
204#else
205        int c =
206            mygetopt(argc, argv, "c:S:R::l::r::hv", long_options,
207                     &option_index);
208#endif
209        if (c == -1)
210            break;
211
212        switch (c)
213        {
214        case 'c':              /* --config */
215            if (screen_list->user_path)
216                free(screen_list->user_path);
217            screen_list->user_path = strdup(myoptarg);
218            s += 2;
219            break;
220        case 'S':
221            if (!screen_list->session_name)
222                screen_list->session_name = strdup(myoptarg);
223            s += 2;
224            break;
225        case 'P':              /* --pid */
226            if (myoptarg)
227            {
228                pidopt = atoi(myoptarg);
229                if (pidopt <= 0)
230                {
231                    fprintf(stderr, "Invalid pid %d\n", pidopt);
232                    if (screen_list->to_grab)
233                        free(screen_list->to_grab);
234                    return -1;
235                }
236            }
237            else
238                pidopt = select_process(screen_list);
239            if (pidopt <= 0)
240            {
241                s += 1;
242                break;
243            }
244            if (!screen_list->to_grab)
245            {
246                /* At most argc-1-s times -P <pid> + final 0 */
247                screen_list->to_grab =
248                    (int *)malloc(((argc - 1 - s) / 2 + 1) * sizeof(int));
249                if (!screen_list->to_grab)
250                {
251                    fprintf(stderr, "Can't allocate memory at %s:%d\n",
252                            __FUNCTION__, __LINE__);
253                    return -1;
254                }
255            }
256            screen_list->to_grab[screen_list->nb_to_grab++] = pidopt;
257            screen_list->to_grab[screen_list->nb_to_grab] = 0;
258            s += 2;
259            break;
260        case 'l':
261            screen_list->autolock_timeout = atoi(myoptarg) * 1000000;
262            if (screen_list->autolock_timeout == 0)
263                screen_list->autolock_timeout -= 1;
264            break;
265        case 'r':
266            screen_list->forceattach = 1;
267        case 'R':
268            if (screen_list->attach)
269            {
270                fprintf(stderr, "Attaching can only be requested once\n");
271                return -1;
272            }
273            if (myoptarg)
274            {
275                if (screen_list->session_name)
276                    free(screen_list->session_name);
277                screen_list->session_name = strdup(myoptarg);
278                s += 1;
279            }
280            screen_list->attach = 1;
281            s += 1;
282            break;
283        case 'h':              /* --help */
284            usage(argc, argv);
285            return -1;
286            break;
287        case 'v':              /* --version */
288            version();
289            return -1;
290            break;
291        case -2:
292            return -1;
293        default:
294            fprintf(stderr, "Unknown argument #%d\n", myoptind);
295            return -1;
296            break;
297        }
298    }
299    if (s >= 0 && s < argc - 1)
300    {
301        screen_list->to_start = (char **)malloc((argc - s) * sizeof(char *));
302        if (!screen_list->to_start)
303        {
304            fprintf(stderr, "Can't allocate memory at %s:%d\n", __FUNCTION__,
305                    __LINE__);
306            return -1;
307        }
308        for (i = 0; i < (argc - 1) - s; i++)
309        {
310            screen_list->to_start[i] = strdup(argv[i + s + 1]);
311        }
312        screen_list->to_start[argc - 1 - s] = NULL;
313    }
314    return s;
315}
Note: See TracBrowser for help on using the repository browser.