]> git.pond.sub.org Git - empserver/blob - src/client/main.c
33c29d98ae12c7b34d87880dc15fe55ad4b4e3ad
[empserver] / src / client / main.c
1 /*
2  *  Empire - A multi-player, client/server Internet based war game.
3  *  Copyright (C) 1986-2011, Dave Pare, Jeff Bailey, Thomas Ruschak,
4  *                Ken Stevens, Steve McClure, Markus Armbruster
5  *
6  *  Empire is free software: you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation, either version 3 of the License, or
9  *  (at your option) any later version.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  *
19  *  ---
20  *
21  *  See files README, COPYING and CREDITS in the root of the source
22  *  tree for related information and legal notices.  It is expected
23  *  that future projects/authors will amend these files as needed.
24  *
25  *  ---
26  *
27  *  main.c: client main function
28  *
29  *  Known contributors to this file:
30  *     Dave Pare, 1986
31  *     Steve McClure, 1998
32  *     Ron Koenderink, 2004-2007
33  *     Markus Armbruster, 2005-2009
34  */
35
36 #include <config.h>
37
38 #include <stdlib.h>
39 #include <string.h>
40 #ifdef _WIN32
41 #include <windows.h>
42 #include "sys/socket.h"
43 #else
44 #include <pwd.h>
45 #endif
46 #include <unistd.h>
47 #include "misc.h"
48 #include "version.h"
49
50 #ifdef _WIN32
51 #define getuid() 0
52 #define getpwuid(uid) ((void)(uid), w32_getpw())
53 #define sysdep_init() w32_sysdep_init()
54
55 struct passwd {
56     char *pw_name;
57 };
58
59 static struct passwd *w32_getpw(void);
60 static void w32_sysdep_init(void);
61 #else  /* !_WIN32 */
62 #define sysdep_init() ((void)0)
63 #endif  /* !_WIN32 */
64
65 static void
66 print_usage(char *program_name)
67 {
68     printf("Usage: %s [OPTION]...[COUNTRY [PASSWORD]]\n"
69            "  -2 FILE         Append log of session to FILE\n"
70            "  -k              Kill connection\n"
71            "  -s [HOST:]PORT  Specify server HOST and PORT\n"
72            "  -u              Use UTF-8\n"
73            "  -h              display this help and exit\n"
74            "  -v              display version information and exit\n",
75            program_name);
76 }
77
78 int
79 main(int argc, char **argv)
80 {
81     int opt;
82     char *auxfname = NULL;
83     int send_kill = 0;
84     char *host = NULL;
85     char *port = NULL;
86     int utf8 = 0;
87     char **ap;
88     char *country;
89     char *passwd;
90     char *uname;
91     char *colon;
92     int sock;
93
94     while ((opt = getopt(argc, argv, "2:ks:uhv")) != EOF) {
95         switch (opt) {
96         case '2':
97             auxfname = optarg;
98             break;
99         case 'k':
100             send_kill = 1;
101             break;
102         case 's':
103             port = strdup(optarg);
104             colon = strrchr(port, ':');
105             if (colon) {
106                 *colon = 0;
107                 host = port;
108                 port = colon + 1;
109             }
110             break;
111         case 'u':
112             utf8 = eight_bit_clean = 1;
113             break;
114         case 'h':
115             print_usage(argv[0]);
116             exit(0);
117         case 'v':
118             printf("%s\n\n%s", version, legal);
119             exit(0);
120         default:
121             fprintf(stderr, "Try -h for help.\n");
122             exit(1);
123         }
124     }
125
126     ap = argv + optind;
127     if (*ap)
128         country = *ap++;
129     else
130         country = getenv("COUNTRY");
131     if (*ap)
132         passwd = *ap++;
133     else
134         passwd = getenv("PLAYER");
135     if (!port)
136         port = getenv("EMPIREPORT");
137     if (!port)
138         port = empireport;
139     if (!host)
140         host = getenv("EMPIREHOST");
141     if (!host)
142         host = empirehost;
143     uname = getenv("LOGNAME");
144     if (uname == NULL) {
145         struct passwd *pwd;
146
147         pwd = getpwuid(getuid());
148         if (pwd == NULL) {
149             fprintf(stderr, "You don't exist.  Go away\n");
150             exit(1);
151         }
152         uname = pwd->pw_name;
153     }
154     if (*ap) {
155         fprintf(stderr, "%s: extra operand %s\n", argv[0], *ap);
156         fprintf(stderr, "Try -h for help.\n");
157         exit(1);
158     }
159
160     getsose();
161     if (auxfname && (auxfp = fopen(auxfname, "a")) == NULL) {
162         fprintf(stderr, "Unable to open %s for append\n", auxfname);
163         exit(1);
164     }
165
166     sysdep_init();
167
168     sock = tcp_connect(host, port);
169
170     if (!login(sock, uname, country, passwd, send_kill, utf8))
171         exit(1);
172
173     if (play(sock) < 0)
174         exit(1);
175
176     return 0;
177 }
178
179 #ifdef _WIN32
180 /*
181  * Get Windows user name
182  */
183 static struct passwd *
184 w32_getpw(void)
185 {
186     static char unamebuf[128];
187     static struct passwd pwd;
188     DWORD unamesize;
189
190     unamesize = sizeof(unamebuf);
191     if (GetUserName(unamebuf, &unamesize)) {
192         pwd.pw_name = unamebuf;
193         if (unamesize == 0 || strlen(unamebuf) == 0)
194             pwd.pw_name = "nobody";
195     } else
196         pwd.pw_name = "nobody";
197     return &pwd;
198 }
199
200 static void
201 w32_sysdep_init(void)
202 {
203     int err;
204
205     /*
206      * stdout is unbuffered under Windows if connected to a character
207      * device, and putchar() screws up when printing multibyte strings
208      * bytewise to an unbuffered stream.  Switch stdout to line-
209      * buffered mode.  Unfortunately, ISO C allows implementations to
210      * screw that up, and of course Windows does.  Manual flushing
211      * after each prompt is required.
212      */
213     setvbuf(stdout, NULL, _IOLBF, 4096);
214
215     err = w32_socket_init();
216     if (err != 0) {
217         fprintf(stderr, "WSAStartup Failed, error code %d\n", err);
218         exit(1);
219     }
220 }
221 #endif