a0e7a23e6a3d51fed3f1c0870cd07559e7e438e1
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.
20 "@(#) Copyright (c) 1985, 1989 Regents of the University of California.\n\
21 All rights reserved.\n";
25 static char sccsid
[] = "@(#)main.c based on 5.13 (Berkeley) 3/14/89";
29 * FTP User Program -- Command Interface.
33 #include <sys/socket.h>
34 #include <sys/ioctl.h>
40 #include "prototypes.h"
41 #include <sys/types.h>
52 #if defined(sun) && !defined(FD_SET)
69 /* Lot's of options... */
71 * Options and other state info.
73 int trace
; /* trace packets exchanged */
74 int hash
; /* print # for each buffer transferred */
75 int sendport
; /* use PORT cmd for each data connection */
76 int verbose
; /* print messages coming back from server */
77 int connected
; /* connected to server */
78 int fromatty
; /* input is from a terminal */
79 int interactive
; /* interactively prompt on m* cmds */
80 int debug
; /* debugging level */
81 int bell
; /* ring bell on cmd completion */
82 int doglob
; /* glob local file names */
83 int proxy
; /* proxy server connection active */
85 int proxflag
; /* proxy connection exists */
86 int sunique
; /* store files on server with unique name */
87 int runique
; /* store local files with unique name */
88 int mcase
; /* map upper to lower case for mget names */
89 int ntflag
; /* use ntin ntout tables for name translation */
90 int mapflag
; /* use mapin mapout templates on file names */
91 int code
; /* return/reply code for ftp command */
92 int crflag
; /* if 1, strip car. rets. on ascii gets */
93 char pasv
[64]; /* passive port for proxy data connection */
94 char *altarg
; /* argv[1] with no shell-like preprocessing */
95 char ntin
[17]; /* input translation table */
96 char ntout
[17]; /* output translation table */
97 // #include <sys/param.h>
98 char mapin
[MAXPATHLEN
]; /* input map template */
99 char mapout
[MAXPATHLEN
]; /* output map template */
100 char typename
[32]; /* name of file transfer type */
101 int type
; /* file transfer type */
102 char structname
[32]; /* name of file transfer structure */
103 int stru
; /* file transfer structure */
104 char formname
[32]; /* name of file transfer format */
105 int form
; /* file transfer format */
106 char modename
[32]; /* name of file transfer mode */
107 int mode
; /* file transfer mode */
108 char bytename
[32]; /* local byte size in ascii */
109 int bytesize
; /* local byte size in binary */
111 jmp_buf toplevel
; /* non-local goto stuff for cmd scanner */
113 char line
[200]; /* input line buffer */
114 char *stringbase
; /* current scan point in line buffer */
115 char argbuf
[200]; /* argument storage buffer */
116 char *argbase
; /* current storage point in arg buffer */
117 int margc
; /* count of arguments on input line */
118 char *margv
[20]; /* args parsed from input line */
119 int cpend
; /* flag: if != 0, then pending server reply */
120 int mflag
; /* flag: if != 0, then active multi command */
122 int options
; /* used during socket creation */
126 int main(int argc
, char *argv
[])
131 char homedir
[MAXPATHLEN
];
138 struct servent
*sp
; /* service spec for tcp/ftp */
140 /* Disable output buffering, for the benefit of Emacs. */
141 //setbuf(stdout, NULL);
143 _fmode
= O_BINARY
; // This causes an error somewhere.
145 wVerReq
= MAKEWORD(1,1);
147 err
= WSAStartup(wVerReq
, &WSAData
);
150 fprintf(stderr
, "Could not initialize Windows socket interface.");
154 sp
= getservbyname("ftp", "tcp");
156 fprintf(stderr
, "ftp: ftp/tcp: unknown service\n");
160 portnum
= sp
->s_port
;
167 while (argc
> 0 && **argv
== '-') {
168 for (cp
= *argv
+ 1; *cp
; cp
++)
198 "ftp: %c: unknown option\n", *cp
);
203 // fromatty = isatty(fileno(stdin));
204 fromatty
= 1; // Strengthen this test
206 * Set up defaults for FTP.
208 (void) strcpy(typename
, "ascii"), type
= TYPE_A
;
209 (void) strcpy(formname
, "non-print"), form
= FORM_N
;
210 (void) strcpy(modename
, "stream"), mode
= MODE_S
;
211 (void) strcpy(structname
, "file"), stru
= STRU_F
;
212 (void) strcpy(bytename
, "8"), bytesize
= 8;
215 cpend
= 0; /* no pending replies */
216 proxy
= 0; /* proxy not active */
217 passivemode
= 1; /* passive mode *is* active */
218 crflag
= 1; /* strip c.r. on ascii gets */
220 * Set up the home directory in case we're globbing.
228 pw
= getpwuid(getuid());
231 (void) strcpy(home
, pw
->pw_dir
);
236 if (setjmp(toplevel
))
238 // (void) signal(SIGINT, intr);
239 // (void) signal(SIGPIPE, lostpeer);
240 setpeer(argc
+ 1, argv
- 1);
242 top
= setjmp(toplevel
) == 0;
244 // (void) signal(SIGINT, intr);
245 // (void) signal(SIGPIPE, lostpeer);
257 longjmp(toplevel
, 1);
266 if (cout
!= (int) NULL
) {
271 (void) shutdown(data
, 1+1);
279 if (cout
!= (int)NULL
) {
296 s = rindex(filename, '/');
312 register struct cmd
*c
;
313 struct cmd
*getcmd();
317 (void) putchar('\n');
319 (void) fflush(stdout
);
322 (void) fflush(stdout
);
324 if (gets(line
) == 0) {
325 if (feof(stdin
) || ferror(stdin
))
335 c
= getcmd(margv
[0]);
336 if (c
== (struct cmd
*)-1) {
337 printf("?Ambiguous command\n");
341 printf("?Invalid command\n");
344 if (c
->c_conn
&& !connected
) {
345 printf ("Not connected.\n");
348 (*c
->c_handler
)(margc
, margv
);
349 if (bell
&& c
->c_bell
)
350 (void) putchar('\007');
351 if (c
->c_handler
!= help
)
354 (void) fflush(stdout
);
355 // (void) signal(SIGINT, intr);
356 // (void) signal(SIGPIPE, lostpeer);
363 extern struct cmd cmdtab
[];
364 register char *p
, *q
;
365 register struct cmd
*c
, *found
;
366 register int nmatches
, longest
;
371 for (c
= cmdtab
; (p
= c
->c_name
); c
++) {
372 for (q
= name
; *q
== *p
++; q
++)
373 if (*q
== 0) /* exact match? */
375 if (!*q
) { /* the name was a prefix */
376 if (q
- name
> longest
) {
380 } else if (q
- name
== longest
)
385 return ((struct cmd
*)-1);
390 * Slice a string up into argc/argv.
402 stringbase
= line
; /* scan from first of buffer */
403 argbase
= argbuf
; /* store from first of buffer */
405 while ((*argp
++ = slurpstring()))
410 * Parse string into argbuf;
411 * implemented with FSM to
412 * handle quoting and strings
418 register char *sb
= stringbase
;
419 register char *ap
= argbase
;
420 char *tmp
= argbase
; /* will return this if token found */
422 if (*sb
== '!' || *sb
== '$') { /* recognize ! as a token for shell */
423 switch (slrflag
) { /* and $ as token for macro invoke */
427 return ((*sb
== '!') ? "!" : "$");
469 goto OUT1
; /* end of token */
472 sb
++; goto S2
; /* slurp next character */
475 sb
++; goto S3
; /* slurp quoted string */
478 *ap
++ = *sb
++; /* add character to token */
513 argbase
= ap
; /* update storage pointer */
514 stringbase
= sb
; /* update scan pointer */
532 #define HELPINDENT (sizeof ("directory"))
536 * Call each command handler with argc == 0 and argv[0] == name.
542 extern struct cmd cmdtab
[];
543 register struct cmd
*c
;
546 register int i
, j
, w
, k
;
547 int columns
, width
= 0, lines
;
550 printf("Commands may be abbreviated. Commands are:\n\n");
551 for (c
= cmdtab
; c
< &cmdtab
[NCMDS
]; c
++) {
552 int len
= strlen(c
->c_name
);
557 width
= (width
+ 8) &~ 7;
558 columns
= 80 / width
;
561 lines
= (NCMDS
+ columns
- 1) / columns
;
562 for (i
= 0; i
< lines
; i
++) {
563 for (j
= 0; j
< columns
; j
++) {
564 c
= cmdtab
+ j
* lines
+ i
;
565 if (c
->c_name
&& (!proxy
|| c
->c_proxy
)) {
566 printf("%s", c
->c_name
);
568 else if (c
->c_name
) {
569 for (k
=0; k
< (int) strlen(c
->c_name
); k
++) {
573 if (c
+ lines
>= &cmdtab
[NCMDS
]) {
577 w
= strlen(c
->c_name
);
580 (void) putchar('\t');
584 (void) fflush(stdout
);
591 if (c
== (struct cmd
*)-1)
592 printf("?Ambiguous help command %s\n", arg
);
593 else if (c
== (struct cmd
*)0)
594 printf("?Invalid help command %s\n", arg
);
596 printf("%-*s\t%s\n", HELPINDENT
,
597 c
->c_name
, c
->c_help
);
599 (void) fflush(stdout
);