2 * Copyright (c) 1985, 1989 Regents of the University of California.
5 * Redistribution and use in source and binary forms are permitted
6 * provided that the above copyright notice and this paragraph are
7 * duplicated in all such forms and that any documentation,
8 * advertising materials, and other materials related to such
9 * distribution and use acknowledge that the software was developed
10 * by the University of California, Berkeley. The name of the
11 * University may not be used to endorse or promote products derived
12 * from this software without specific prior written permission.
13 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
14 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
15 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
19 * FTP User Program -- Command Interface.
28 "@(#) Copyright (c) 1985, 1989 Regents of the University of California.\n\
29 All rights reserved.\n";
33 static char sccsid
[] = "@(#)main.c based on 5.13 (Berkeley) 3/14/89";
36 #if defined(sun) && !defined(FD_SET)
53 /* Lot's of options... */
55 * Options and other state info.
57 int trace
; /* trace packets exchanged */
58 int hash
; /* print # for each buffer transferred */
59 //int sendport; /* use PORT cmd for each data connection */
60 int verbose
; /* print messages coming back from server */
61 int connected
; /* connected to server */
62 int fromatty
; /* input is from a terminal */
63 int interactive
; /* interactively prompt on m* cmds */
64 int debug
; /* debugging level */
65 int bell
; /* ring bell on cmd completion */
66 int doglob
; /* glob local file names */
67 int proxy
; /* proxy server connection active */
69 int proxflag
; /* proxy connection exists */
70 int sunique
; /* store files on server with unique name */
71 int runique
; /* store local files with unique name */
72 int mcase
; /* map upper to lower case for mget names */
73 int ntflag
; /* use ntin ntout tables for name translation */
74 int mapflag
; /* use mapin mapout templates on file names */
75 int code
; /* return/reply code for ftp command */
76 int crflag
; /* if 1, strip car. rets. on ascii gets */
77 char pasv
[64]; /* passive port for proxy data connection */
78 char *altarg
; /* argv[1] with no shell-like preprocessing */
79 char ntin
[17]; /* input translation table */
80 char ntout
[17]; /* output translation table */
81 // #include <sys/param.h>
82 char mapin
[MAXPATHLEN
]; /* input map template */
83 char mapout
[MAXPATHLEN
]; /* output map template */
84 char typename
[32]; /* name of file transfer type */
85 int type
; /* file transfer type */
86 char structname
[32]; /* name of file transfer structure */
87 int stru
; /* file transfer structure */
88 char formname
[32]; /* name of file transfer format */
89 int form
; /* file transfer format */
90 char modename
[32]; /* name of file transfer mode */
91 int mode
; /* file transfer mode */
92 char bytename
[32]; /* local byte size in ascii */
93 int bytesize
; /* local byte size in binary */
95 jmp_buf toplevel
; /* non-local goto stuff for cmd scanner */
97 char line
[200]; /* input line buffer */
98 char *stringbase
; /* current scan point in line buffer */
99 char argbuf
[200]; /* argument storage buffer */
100 char *argbase
; /* current storage point in arg buffer */
101 int margc
; /* count of arguments on input line */
102 const char *margv
[20]; /* args parsed from input line */
103 int cpend
; /* flag: if != 0, then pending server reply */
104 int mflag
; /* flag: if != 0, then active multi command */
106 int options
; /* used during socket creation */
108 int macnum
; /* number of defined macros */
109 struct macel macros
[16];
113 * Need to start a listen on the data channel
114 * before we send the command, otherwise the
115 * server's connect may fail.
119 static const char *slurpstring();
122 int main(int argc
, const char *argv
[])
127 char homedir
[MAXPATHLEN
];
134 struct servent
*sp
; /* service spec for tcp/ftp */
136 /* Disable output buffering, for the benefit of Emacs. */
137 //setbuf(stdout, NULL);
139 _fmode
= O_BINARY
; // This causes an error somewhere.
141 wVerReq
= MAKEWORD(1,1);
143 err
= WSAStartup(wVerReq
, &WSAData
);
146 fprintf(stderr
, "Could not initialize Windows socket interface.");
150 sp
= getservbyname("ftp", "tcp");
152 fprintf(stderr
, "ftp: ftp/tcp: unknown service\n");
156 portnum
= sp
->s_port
;
163 while (argc
> 0 && **argv
== '-') {
164 for (cp
= *argv
+ 1; *cp
; cp
++)
194 "ftp: %c: unknown option\n", *cp
);
199 // fromatty = isatty(fileno(stdin));
200 fromatty
= 1; // Strengthen this test
202 * Set up defaults for FTP.
204 (void) strcpy(typename
, "ascii"), type
= TYPE_A
;
205 (void) strcpy(formname
, "non-print"), form
= FORM_N
;
206 (void) strcpy(modename
, "stream"), mode
= MODE_S
;
207 (void) strcpy(structname
, "file"), stru
= STRU_F
;
208 (void) strcpy(bytename
, "8"), bytesize
= 8;
211 cpend
= 0; /* no pending replies */
212 proxy
= 0; /* proxy not active */
213 passivemode
= 1; /* passive mode *is* active */
214 crflag
= 1; /* strip c.r. on ascii gets */
216 * Set up the home directory in case we're globbing.
224 pw
= getpwuid(getuid());
227 (void) strcpy(home
, pw
->pw_dir
);
232 if (setjmp(toplevel
))
234 // (void) signal(SIGINT, intr);
235 // (void) signal(SIGPIPE, lostpeer);
236 setpeer(argc
+ 1, argv
- 1);
238 top
= setjmp(toplevel
) == 0;
240 // (void) signal(SIGINT, intr);
241 // (void) signal(SIGPIPE, lostpeer);
251 longjmp(toplevel
, 1);
265 (void) shutdown(data
, 1+1);
289 s = rindex(filename, '/');
302 void cmdscanner(int top
)
304 register struct cmd
*c
;
307 (void) putchar('\n');
309 (void) fflush(stdout
);
312 (void) fflush(stdout
);
314 if (gets(line
) == 0) {
315 if (feof(stdin
) || ferror(stdin
))
325 c
= getcmd(margv
[0]);
326 if (c
== (struct cmd
*)-1) {
327 printf("?Ambiguous command\n");
331 printf("?Invalid command\n");
334 if (c
->c_conn
&& !connected
) {
335 printf ("Not connected.\n");
338 (*c
->c_handler
)(margc
, margv
);
339 if (bell
&& c
->c_bell
)
340 (void) putchar('\007');
341 if (c
->c_handler
!= help
)
344 (void) fflush(stdout
);
345 // (void) signal(SIGINT, intr);
346 // (void) signal(SIGPIPE, lostpeer);
350 getcmd(const char *name
)
352 extern struct cmd cmdtab
[];
354 struct cmd
*c
, *found
;
355 int nmatches
, longest
;
360 for (c
= cmdtab
; (p
= c
->c_name
); c
++) {
361 for (q
= name
; *q
== *p
++; q
++)
362 if (*q
== 0) /* exact match? */
364 if (!*q
) { /* the name was a prefix */
365 if (q
- name
> longest
) {
369 } else if (q
- name
== longest
)
374 return ((struct cmd
*)-1);
379 * Slice a string up into argc/argv.
390 stringbase
= line
; /* scan from first of buffer */
391 argbase
= argbuf
; /* store from first of buffer */
393 while ((*argp
++ = slurpstring()))
398 * Parse string into argbuf;
399 * implemented with FSM to
400 * handle quoting and strings
406 register char *sb
= stringbase
;
407 register char *ap
= argbase
;
408 char *tmp
= argbase
; /* will return this if token found */
410 if (*sb
== '!' || *sb
== '$') { /* recognize ! as a token for shell */
411 switch (slrflag
) { /* and $ as token for macro invoke */
415 return ((*sb
== '!') ? "!" : "$");
457 goto OUT1
; /* end of token */
460 sb
++; goto S2
; /* slurp next character */
463 sb
++; goto S3
; /* slurp quoted string */
466 *ap
++ = *sb
++; /* add character to token */
501 argbase
= ap
; /* update storage pointer */
502 stringbase
= sb
; /* update scan pointer */
520 #define HELPINDENT (sizeof ("directory"))
524 * Call each command handler with argc == 0 and argv[0] == name.
526 void help(int argc
, const char *argv
[])
528 extern struct cmd cmdtab
[];
532 register int i
, j
, w
, k
;
533 int columns
, width
= 0, lines
;
536 printf("Commands may be abbreviated. Commands are:\n\n");
537 for (c
= cmdtab
; c
< &cmdtab
[NCMDS
]; c
++) {
538 int len
= strlen(c
->c_name
);
543 width
= (width
+ 8) &~ 7;
544 columns
= 80 / width
;
547 lines
= (NCMDS
+ columns
- 1) / columns
;
548 for (i
= 0; i
< lines
; i
++) {
549 for (j
= 0; j
< columns
; j
++) {
550 c
= cmdtab
+ j
* lines
+ i
;
551 if (c
->c_name
&& (!proxy
|| c
->c_proxy
)) {
552 printf("%s", c
->c_name
);
554 else if (c
->c_name
) {
555 for (k
=0; k
< (int) strlen(c
->c_name
); k
++) {
559 if (c
+ lines
>= &cmdtab
[NCMDS
]) {
563 w
= strlen(c
->c_name
);
566 (void) putchar('\t');
570 (void) fflush(stdout
);
577 if (c
== (struct cmd
*)-1)
578 printf("?Ambiguous help command %s\n", arg
);
579 else if (c
== (struct cmd
*)0)
580 printf("?Invalid help command %s\n", arg
);
582 printf("%-*s\t%s\n", (int)HELPINDENT
,
583 c
->c_name
, c
->c_help
);
585 (void) fflush(stdout
);