2003-09-02 09:00:35 +00:00
|
|
|
/*
|
|
|
|
* psftp.h: interface between psftp.c / scp.c and each
|
|
|
|
* platform-specific SFTP module.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef PUTTY_PSFTP_H
|
|
|
|
#define PUTTY_PSFTP_H
|
|
|
|
|
|
|
|
/*
|
|
|
|
* psftp_getcwd returns the local current directory. The returned
|
|
|
|
* string must be freed by the caller.
|
|
|
|
*/
|
|
|
|
char *psftp_getcwd(void);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* psftp_lcd changes the local current directory. The return value
|
|
|
|
* is NULL on success, or else an error message which must be freed
|
|
|
|
* by the caller.
|
|
|
|
*/
|
|
|
|
char *psftp_lcd(char *newdir);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Retrieve file times on a local file. Must return two unsigned
|
|
|
|
* longs in POSIX time_t format.
|
|
|
|
*/
|
|
|
|
void get_file_times(char *filename, unsigned long *mtime,
|
|
|
|
unsigned long *atime);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* One iteration of the PSFTP event loop: wait for network data and
|
|
|
|
* process it, once.
|
|
|
|
*/
|
|
|
|
int ssh_sftp_loop_iteration(void);
|
|
|
|
|
2004-11-27 13:20:21 +00:00
|
|
|
/*
|
|
|
|
* Read a command line for PSFTP from standard input. Caller must
|
|
|
|
* free.
|
2004-12-16 19:15:38 +00:00
|
|
|
*
|
2018-10-29 19:50:29 +00:00
|
|
|
* If `backend_required' is true, should also listen for activity
|
2004-12-16 19:15:38 +00:00
|
|
|
* at the backend (rekeys, clientalives, unexpected closures etc)
|
|
|
|
* and respond as necessary, and if the backend closes it should
|
|
|
|
* treat this as a failure condition. If `backend_required' is
|
2018-10-29 19:50:29 +00:00
|
|
|
* false, a back end is not (intentionally) active at all (e.g.
|
2004-12-16 19:15:38 +00:00
|
|
|
* psftp before an `open' command).
|
2004-11-27 13:20:21 +00:00
|
|
|
*/
|
Convert a lot of 'int' variables to 'bool'.
My normal habit these days, in new code, is to treat int and bool as
_almost_ completely separate types. I'm still willing to use C's
implicit test for zero on an integer (e.g. 'if (!blob.len)' is fine,
no need to spell it out as blob.len != 0), but generally, if a
variable is going to be conceptually a boolean, I like to declare it
bool and assign to it using 'true' or 'false' rather than 0 or 1.
PuTTY is an exception, because it predates the C99 bool, and I've
stuck to its existing coding style even when adding new code to it.
But it's been annoying me more and more, so now that I've decided C99
bool is an acceptable thing to require from our toolchain in the first
place, here's a quite thorough trawl through the source doing
'boolification'. Many variables and function parameters are now typed
as bool rather than int; many assignments of 0 or 1 to those variables
are now spelled 'true' or 'false'.
I managed this thorough conversion with the help of a custom clang
plugin that I wrote to trawl the AST and apply heuristics to point out
where things might want changing. So I've even managed to do a decent
job on parts of the code I haven't looked at in years!
To make the plugin's work easier, I pushed platform front ends
generally in the direction of using standard 'bool' in preference to
platform-specific boolean types like Windows BOOL or GTK's gboolean;
I've left the platform booleans in places they _have_ to be for the
platform APIs to work right, but variables only used by my own code
have been converted wherever I found them.
In a few places there are int values that look very like booleans in
_most_ of the places they're used, but have a rarely-used third value,
or a distinction between different nonzero values that most users
don't care about. In these cases, I've _removed_ uses of 'true' and
'false' for the return values, to emphasise that there's something
more subtle going on than a simple boolean answer:
- the 'multisel' field in dialog.h's list box structure, for which
the GTK front end in particular recognises a difference between 1
and 2 but nearly everything else treats as boolean
- the 'urgent' parameter to plug_receive, where 1 vs 2 tells you
something about the specific location of the urgent pointer, but
most clients only care about 0 vs 'something nonzero'
- the return value of wc_match, where -1 indicates a syntax error in
the wildcard.
- the return values from SSH-1 RSA-key loading functions, which use
-1 for 'wrong passphrase' and 0 for all other failures (so any
caller which already knows it's not loading an _encrypted private_
key can treat them as boolean)
- term->esc_query, and the 'query' parameter in toggle_mode in
terminal.c, which _usually_ hold 0 for ESC[123h or 1 for ESC[?123h,
but can also hold -1 for some other intervening character that we
don't support.
In a few places there's an integer that I haven't turned into a bool
even though it really _can_ only take values 0 or 1 (and, as above,
tried to make the call sites consistent in not calling those values
true and false), on the grounds that I thought it would make it more
confusing to imply that the 0 value was in some sense 'negative' or
bad and the 1 positive or good:
- the return value of plug_accepting uses the POSIXish convention of
0=success and nonzero=error; I think if I made it bool then I'd
also want to reverse its sense, and that's a job for a separate
piece of work.
- the 'screen' parameter to lineptr() in terminal.c, where 0 and 1
represent the default and alternate screens. There's no obvious
reason why one of those should be considered 'true' or 'positive'
or 'success' - they're just indices - so I've left it as int.
ssh_scp_recv had particularly confusing semantics for its previous int
return value: its call sites used '<= 0' to check for error, but it
never actually returned a negative number, just 0 or 1. Now the
function and its call sites agree that it's a bool.
In a couple of places I've renamed variables called 'ret', because I
don't like that name any more - it's unclear whether it means the
return value (in preparation) for the _containing_ function or the
return value received from a subroutine call, and occasionally I've
accidentally used the same variable for both and introduced a bug. So
where one of those got in my way, I've renamed it to 'toret' or 'retd'
(the latter short for 'returned') in line with my usual modern
practice, but I haven't done a thorough job of finding all of them.
Finally, one amusing side effect of doing this is that I've had to
separate quite a few chained assignments. It used to be perfectly fine
to write 'a = b = c = TRUE' when a,b,c were int and TRUE was just a
the 'true' defined by stdbool.h, that idiom provokes a warning from
gcc: 'suggest parentheses around assignment used as truth value'!
2018-11-02 19:23:19 +00:00
|
|
|
char *ssh_sftp_get_cmdline(const char *prompt, bool backend_required);
|
2004-11-27 13:20:21 +00:00
|
|
|
|
2016-04-02 07:00:07 +00:00
|
|
|
/*
|
2017-02-11 00:44:00 +00:00
|
|
|
* Platform-specific function called when we're about to make a
|
|
|
|
* network connection.
|
2016-04-02 07:00:07 +00:00
|
|
|
*/
|
2017-02-11 00:44:00 +00:00
|
|
|
void platform_psftp_pre_conn_setup(void);
|
2016-04-02 07:00:07 +00:00
|
|
|
|
2003-09-02 09:00:35 +00:00
|
|
|
/*
|
|
|
|
* The main program in psftp.c. Called from main() in the platform-
|
|
|
|
* specific code, after doing any platform-specific initialisation.
|
|
|
|
*/
|
|
|
|
int psftp_main(int argc, char *argv[]);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These functions are used by PSCP to transmit progress updates
|
|
|
|
* and error information to a GUI window managing it. This will
|
|
|
|
* probably only ever be supported on Windows, so these functions
|
|
|
|
* can safely be stubs on all other platforms.
|
|
|
|
*/
|
2015-05-15 10:15:42 +00:00
|
|
|
void gui_update_stats(const char *name, unsigned long size,
|
2003-09-02 09:00:35 +00:00
|
|
|
int percentage, unsigned long elapsed,
|
|
|
|
unsigned long done, unsigned long eta,
|
|
|
|
unsigned long ratebs);
|
|
|
|
void gui_send_errcount(int list, int errs);
|
|
|
|
void gui_send_char(int is_stderr, int c);
|
2015-05-15 10:15:42 +00:00
|
|
|
void gui_enable(const char *arg);
|
2003-09-02 09:00:35 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* It's likely that a given platform's implementation of file
|
|
|
|
* transfer utilities is going to want to do things with them that
|
|
|
|
* aren't present in stdio. Hence we supply an alternative
|
|
|
|
* abstraction for file access functions.
|
|
|
|
*
|
|
|
|
* This abstraction tells you the size and access times when you
|
|
|
|
* open an existing file (platforms may choose the meaning of the
|
|
|
|
* file times if it's not clear; whatever they choose will be what
|
|
|
|
* PSCP sends to the server as mtime and atime), and lets you set
|
|
|
|
* the times when saving a new file.
|
|
|
|
*
|
|
|
|
* On the other hand, the abstraction is pretty simple: it supports
|
2006-08-12 15:20:19 +00:00
|
|
|
* only opening a file and reading it, or creating a file and writing
|
|
|
|
* it. None of this read-and-write, seeking-back-and-forth stuff.
|
2003-09-02 09:00:35 +00:00
|
|
|
*/
|
|
|
|
typedef struct RFile RFile;
|
|
|
|
typedef struct WFile WFile;
|
2011-08-11 17:59:30 +00:00
|
|
|
/* Output params size, perms, mtime and atime can all be NULL if
|
|
|
|
* desired. perms will be -1 if the OS does not support POSIX permissions. */
|
2018-10-26 22:08:58 +00:00
|
|
|
RFile *open_existing_file(const char *name, uint64_t *size,
|
2011-08-11 17:59:30 +00:00
|
|
|
unsigned long *mtime, unsigned long *atime,
|
|
|
|
long *perms);
|
2018-10-26 22:08:58 +00:00
|
|
|
WFile *open_existing_wfile(const char *name, uint64_t *size);
|
2003-09-02 09:00:35 +00:00
|
|
|
/* Returns <0 on error, 0 on eof, or number of bytes read, as usual */
|
|
|
|
int read_from_file(RFile *f, void *buffer, int length);
|
|
|
|
/* Closes and frees the RFile */
|
|
|
|
void close_rfile(RFile *f);
|
2015-05-15 10:15:42 +00:00
|
|
|
WFile *open_new_file(const char *name, long perms);
|
2003-09-02 09:00:35 +00:00
|
|
|
/* Returns <0 on error, 0 on eof, or number of bytes written, as usual */
|
|
|
|
int write_to_file(WFile *f, void *buffer, int length);
|
|
|
|
void set_file_times(WFile *f, unsigned long mtime, unsigned long atime);
|
|
|
|
/* Closes and frees the WFile */
|
|
|
|
void close_wfile(WFile *f);
|
2006-08-12 15:20:19 +00:00
|
|
|
/* Seek offset bytes through file */
|
|
|
|
enum { FROM_START, FROM_CURRENT, FROM_END };
|
2018-10-26 22:08:58 +00:00
|
|
|
int seek_file(WFile *f, uint64_t offset, int whence);
|
2006-08-12 15:20:19 +00:00
|
|
|
/* Get file position */
|
2018-10-26 22:08:58 +00:00
|
|
|
uint64_t get_file_posn(WFile *f);
|
2003-09-02 09:00:35 +00:00
|
|
|
/*
|
|
|
|
* Determine the type of a file: nonexistent, file, directory or
|
|
|
|
* weird. `weird' covers anything else - named pipes, Unix sockets,
|
|
|
|
* device files, fish, badgers, you name it. Things marked `weird'
|
|
|
|
* will be skipped over in recursive file transfers, so the only
|
|
|
|
* real reason for not lumping them in with `nonexistent' is that
|
|
|
|
* it allows a slightly more sane error message.
|
|
|
|
*/
|
|
|
|
enum {
|
|
|
|
FILE_TYPE_NONEXISTENT, FILE_TYPE_FILE, FILE_TYPE_DIRECTORY, FILE_TYPE_WEIRD
|
|
|
|
};
|
2015-05-15 10:15:42 +00:00
|
|
|
int file_type(const char *name);
|
2003-09-02 09:00:35 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Read all the file names out of a directory.
|
|
|
|
*/
|
|
|
|
typedef struct DirHandle DirHandle;
|
2015-05-15 10:15:42 +00:00
|
|
|
DirHandle *open_directory(const char *name);
|
2003-09-02 09:00:35 +00:00
|
|
|
/* The string returned from this will need freeing if not NULL */
|
|
|
|
char *read_filename(DirHandle *dir);
|
|
|
|
void close_directory(DirHandle *dir);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Test a filespec to see whether it's a local wildcard or not.
|
|
|
|
* Return values:
|
|
|
|
*
|
|
|
|
* - WCTYPE_WILDCARD (this is a wildcard).
|
|
|
|
* - WCTYPE_FILENAME (this is a single file name).
|
|
|
|
* - WCTYPE_NONEXISTENT (whichever it was, nothing of that name exists).
|
|
|
|
*
|
|
|
|
* Some platforms may choose not to support local wildcards when
|
|
|
|
* they come from the command line; in this case they simply never
|
|
|
|
* return WCTYPE_WILDCARD, but still test the file's existence.
|
|
|
|
* (However, all platforms will probably want to support wildcards
|
|
|
|
* inside the PSFTP CLI.)
|
|
|
|
*/
|
|
|
|
enum {
|
|
|
|
WCTYPE_NONEXISTENT, WCTYPE_FILENAME, WCTYPE_WILDCARD
|
|
|
|
};
|
Convert a lot of 'int' variables to 'bool'.
My normal habit these days, in new code, is to treat int and bool as
_almost_ completely separate types. I'm still willing to use C's
implicit test for zero on an integer (e.g. 'if (!blob.len)' is fine,
no need to spell it out as blob.len != 0), but generally, if a
variable is going to be conceptually a boolean, I like to declare it
bool and assign to it using 'true' or 'false' rather than 0 or 1.
PuTTY is an exception, because it predates the C99 bool, and I've
stuck to its existing coding style even when adding new code to it.
But it's been annoying me more and more, so now that I've decided C99
bool is an acceptable thing to require from our toolchain in the first
place, here's a quite thorough trawl through the source doing
'boolification'. Many variables and function parameters are now typed
as bool rather than int; many assignments of 0 or 1 to those variables
are now spelled 'true' or 'false'.
I managed this thorough conversion with the help of a custom clang
plugin that I wrote to trawl the AST and apply heuristics to point out
where things might want changing. So I've even managed to do a decent
job on parts of the code I haven't looked at in years!
To make the plugin's work easier, I pushed platform front ends
generally in the direction of using standard 'bool' in preference to
platform-specific boolean types like Windows BOOL or GTK's gboolean;
I've left the platform booleans in places they _have_ to be for the
platform APIs to work right, but variables only used by my own code
have been converted wherever I found them.
In a few places there are int values that look very like booleans in
_most_ of the places they're used, but have a rarely-used third value,
or a distinction between different nonzero values that most users
don't care about. In these cases, I've _removed_ uses of 'true' and
'false' for the return values, to emphasise that there's something
more subtle going on than a simple boolean answer:
- the 'multisel' field in dialog.h's list box structure, for which
the GTK front end in particular recognises a difference between 1
and 2 but nearly everything else treats as boolean
- the 'urgent' parameter to plug_receive, where 1 vs 2 tells you
something about the specific location of the urgent pointer, but
most clients only care about 0 vs 'something nonzero'
- the return value of wc_match, where -1 indicates a syntax error in
the wildcard.
- the return values from SSH-1 RSA-key loading functions, which use
-1 for 'wrong passphrase' and 0 for all other failures (so any
caller which already knows it's not loading an _encrypted private_
key can treat them as boolean)
- term->esc_query, and the 'query' parameter in toggle_mode in
terminal.c, which _usually_ hold 0 for ESC[123h or 1 for ESC[?123h,
but can also hold -1 for some other intervening character that we
don't support.
In a few places there's an integer that I haven't turned into a bool
even though it really _can_ only take values 0 or 1 (and, as above,
tried to make the call sites consistent in not calling those values
true and false), on the grounds that I thought it would make it more
confusing to imply that the 0 value was in some sense 'negative' or
bad and the 1 positive or good:
- the return value of plug_accepting uses the POSIXish convention of
0=success and nonzero=error; I think if I made it bool then I'd
also want to reverse its sense, and that's a job for a separate
piece of work.
- the 'screen' parameter to lineptr() in terminal.c, where 0 and 1
represent the default and alternate screens. There's no obvious
reason why one of those should be considered 'true' or 'positive'
or 'success' - they're just indices - so I've left it as int.
ssh_scp_recv had particularly confusing semantics for its previous int
return value: its call sites used '<= 0' to check for error, but it
never actually returned a negative number, just 0 or 1. Now the
function and its call sites agree that it's a bool.
In a couple of places I've renamed variables called 'ret', because I
don't like that name any more - it's unclear whether it means the
return value (in preparation) for the _containing_ function or the
return value received from a subroutine call, and occasionally I've
accidentally used the same variable for both and introduced a bug. So
where one of those got in my way, I've renamed it to 'toret' or 'retd'
(the latter short for 'returned') in line with my usual modern
practice, but I haven't done a thorough job of finding all of them.
Finally, one amusing side effect of doing this is that I've had to
separate quite a few chained assignments. It used to be perfectly fine
to write 'a = b = c = TRUE' when a,b,c were int and TRUE was just a
the 'true' defined by stdbool.h, that idiom provokes a warning from
gcc: 'suggest parentheses around assignment used as truth value'!
2018-11-02 19:23:19 +00:00
|
|
|
int test_wildcard(const char *name, bool cmdline);
|
2003-09-02 09:00:35 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Actually return matching file names for a local wildcard.
|
|
|
|
*/
|
|
|
|
typedef struct WildcardMatcher WildcardMatcher;
|
2015-05-15 10:15:42 +00:00
|
|
|
WildcardMatcher *begin_wildcard_matching(const char *name);
|
2003-09-02 09:00:35 +00:00
|
|
|
/* The string returned from this will need freeing if not NULL */
|
|
|
|
char *wildcard_get_filename(WildcardMatcher *dir);
|
|
|
|
void finish_wildcard_matching(WildcardMatcher *dir);
|
|
|
|
|
2004-12-16 19:36:47 +00:00
|
|
|
/*
|
|
|
|
* Vet a filename returned from the remote host, to ensure it isn't
|
|
|
|
* in some way malicious. The idea is that this function is applied
|
|
|
|
* to filenames returned from FXP_READDIR, which means we can panic
|
|
|
|
* if we see _anything_ resembling a directory separator.
|
|
|
|
*
|
2018-10-29 19:50:29 +00:00
|
|
|
* Returns true if the filename is kosher, false if dangerous.
|
2004-12-16 19:36:47 +00:00
|
|
|
*/
|
Convert a lot of 'int' variables to 'bool'.
My normal habit these days, in new code, is to treat int and bool as
_almost_ completely separate types. I'm still willing to use C's
implicit test for zero on an integer (e.g. 'if (!blob.len)' is fine,
no need to spell it out as blob.len != 0), but generally, if a
variable is going to be conceptually a boolean, I like to declare it
bool and assign to it using 'true' or 'false' rather than 0 or 1.
PuTTY is an exception, because it predates the C99 bool, and I've
stuck to its existing coding style even when adding new code to it.
But it's been annoying me more and more, so now that I've decided C99
bool is an acceptable thing to require from our toolchain in the first
place, here's a quite thorough trawl through the source doing
'boolification'. Many variables and function parameters are now typed
as bool rather than int; many assignments of 0 or 1 to those variables
are now spelled 'true' or 'false'.
I managed this thorough conversion with the help of a custom clang
plugin that I wrote to trawl the AST and apply heuristics to point out
where things might want changing. So I've even managed to do a decent
job on parts of the code I haven't looked at in years!
To make the plugin's work easier, I pushed platform front ends
generally in the direction of using standard 'bool' in preference to
platform-specific boolean types like Windows BOOL or GTK's gboolean;
I've left the platform booleans in places they _have_ to be for the
platform APIs to work right, but variables only used by my own code
have been converted wherever I found them.
In a few places there are int values that look very like booleans in
_most_ of the places they're used, but have a rarely-used third value,
or a distinction between different nonzero values that most users
don't care about. In these cases, I've _removed_ uses of 'true' and
'false' for the return values, to emphasise that there's something
more subtle going on than a simple boolean answer:
- the 'multisel' field in dialog.h's list box structure, for which
the GTK front end in particular recognises a difference between 1
and 2 but nearly everything else treats as boolean
- the 'urgent' parameter to plug_receive, where 1 vs 2 tells you
something about the specific location of the urgent pointer, but
most clients only care about 0 vs 'something nonzero'
- the return value of wc_match, where -1 indicates a syntax error in
the wildcard.
- the return values from SSH-1 RSA-key loading functions, which use
-1 for 'wrong passphrase' and 0 for all other failures (so any
caller which already knows it's not loading an _encrypted private_
key can treat them as boolean)
- term->esc_query, and the 'query' parameter in toggle_mode in
terminal.c, which _usually_ hold 0 for ESC[123h or 1 for ESC[?123h,
but can also hold -1 for some other intervening character that we
don't support.
In a few places there's an integer that I haven't turned into a bool
even though it really _can_ only take values 0 or 1 (and, as above,
tried to make the call sites consistent in not calling those values
true and false), on the grounds that I thought it would make it more
confusing to imply that the 0 value was in some sense 'negative' or
bad and the 1 positive or good:
- the return value of plug_accepting uses the POSIXish convention of
0=success and nonzero=error; I think if I made it bool then I'd
also want to reverse its sense, and that's a job for a separate
piece of work.
- the 'screen' parameter to lineptr() in terminal.c, where 0 and 1
represent the default and alternate screens. There's no obvious
reason why one of those should be considered 'true' or 'positive'
or 'success' - they're just indices - so I've left it as int.
ssh_scp_recv had particularly confusing semantics for its previous int
return value: its call sites used '<= 0' to check for error, but it
never actually returned a negative number, just 0 or 1. Now the
function and its call sites agree that it's a bool.
In a couple of places I've renamed variables called 'ret', because I
don't like that name any more - it's unclear whether it means the
return value (in preparation) for the _containing_ function or the
return value received from a subroutine call, and occasionally I've
accidentally used the same variable for both and introduced a bug. So
where one of those got in my way, I've renamed it to 'toret' or 'retd'
(the latter short for 'returned') in line with my usual modern
practice, but I haven't done a thorough job of finding all of them.
Finally, one amusing side effect of doing this is that I've had to
separate quite a few chained assignments. It used to be perfectly fine
to write 'a = b = c = TRUE' when a,b,c were int and TRUE was just a
the 'true' defined by stdbool.h, that idiom provokes a warning from
gcc: 'suggest parentheses around assignment used as truth value'!
2018-11-02 19:23:19 +00:00
|
|
|
bool vet_filename(const char *name);
|
2004-12-16 19:36:47 +00:00
|
|
|
|
2003-09-02 09:00:35 +00:00
|
|
|
/*
|
Convert a lot of 'int' variables to 'bool'.
My normal habit these days, in new code, is to treat int and bool as
_almost_ completely separate types. I'm still willing to use C's
implicit test for zero on an integer (e.g. 'if (!blob.len)' is fine,
no need to spell it out as blob.len != 0), but generally, if a
variable is going to be conceptually a boolean, I like to declare it
bool and assign to it using 'true' or 'false' rather than 0 or 1.
PuTTY is an exception, because it predates the C99 bool, and I've
stuck to its existing coding style even when adding new code to it.
But it's been annoying me more and more, so now that I've decided C99
bool is an acceptable thing to require from our toolchain in the first
place, here's a quite thorough trawl through the source doing
'boolification'. Many variables and function parameters are now typed
as bool rather than int; many assignments of 0 or 1 to those variables
are now spelled 'true' or 'false'.
I managed this thorough conversion with the help of a custom clang
plugin that I wrote to trawl the AST and apply heuristics to point out
where things might want changing. So I've even managed to do a decent
job on parts of the code I haven't looked at in years!
To make the plugin's work easier, I pushed platform front ends
generally in the direction of using standard 'bool' in preference to
platform-specific boolean types like Windows BOOL or GTK's gboolean;
I've left the platform booleans in places they _have_ to be for the
platform APIs to work right, but variables only used by my own code
have been converted wherever I found them.
In a few places there are int values that look very like booleans in
_most_ of the places they're used, but have a rarely-used third value,
or a distinction between different nonzero values that most users
don't care about. In these cases, I've _removed_ uses of 'true' and
'false' for the return values, to emphasise that there's something
more subtle going on than a simple boolean answer:
- the 'multisel' field in dialog.h's list box structure, for which
the GTK front end in particular recognises a difference between 1
and 2 but nearly everything else treats as boolean
- the 'urgent' parameter to plug_receive, where 1 vs 2 tells you
something about the specific location of the urgent pointer, but
most clients only care about 0 vs 'something nonzero'
- the return value of wc_match, where -1 indicates a syntax error in
the wildcard.
- the return values from SSH-1 RSA-key loading functions, which use
-1 for 'wrong passphrase' and 0 for all other failures (so any
caller which already knows it's not loading an _encrypted private_
key can treat them as boolean)
- term->esc_query, and the 'query' parameter in toggle_mode in
terminal.c, which _usually_ hold 0 for ESC[123h or 1 for ESC[?123h,
but can also hold -1 for some other intervening character that we
don't support.
In a few places there's an integer that I haven't turned into a bool
even though it really _can_ only take values 0 or 1 (and, as above,
tried to make the call sites consistent in not calling those values
true and false), on the grounds that I thought it would make it more
confusing to imply that the 0 value was in some sense 'negative' or
bad and the 1 positive or good:
- the return value of plug_accepting uses the POSIXish convention of
0=success and nonzero=error; I think if I made it bool then I'd
also want to reverse its sense, and that's a job for a separate
piece of work.
- the 'screen' parameter to lineptr() in terminal.c, where 0 and 1
represent the default and alternate screens. There's no obvious
reason why one of those should be considered 'true' or 'positive'
or 'success' - they're just indices - so I've left it as int.
ssh_scp_recv had particularly confusing semantics for its previous int
return value: its call sites used '<= 0' to check for error, but it
never actually returned a negative number, just 0 or 1. Now the
function and its call sites agree that it's a bool.
In a couple of places I've renamed variables called 'ret', because I
don't like that name any more - it's unclear whether it means the
return value (in preparation) for the _containing_ function or the
return value received from a subroutine call, and occasionally I've
accidentally used the same variable for both and introduced a bug. So
where one of those got in my way, I've renamed it to 'toret' or 'retd'
(the latter short for 'returned') in line with my usual modern
practice, but I haven't done a thorough job of finding all of them.
Finally, one amusing side effect of doing this is that I've had to
separate quite a few chained assignments. It used to be perfectly fine
to write 'a = b = c = TRUE' when a,b,c were int and TRUE was just a
the 'true' defined by stdbool.h, that idiom provokes a warning from
gcc: 'suggest parentheses around assignment used as truth value'!
2018-11-02 19:23:19 +00:00
|
|
|
* Create a directory. Returns true on success, false on error.
|
2003-09-02 09:00:35 +00:00
|
|
|
*/
|
Convert a lot of 'int' variables to 'bool'.
My normal habit these days, in new code, is to treat int and bool as
_almost_ completely separate types. I'm still willing to use C's
implicit test for zero on an integer (e.g. 'if (!blob.len)' is fine,
no need to spell it out as blob.len != 0), but generally, if a
variable is going to be conceptually a boolean, I like to declare it
bool and assign to it using 'true' or 'false' rather than 0 or 1.
PuTTY is an exception, because it predates the C99 bool, and I've
stuck to its existing coding style even when adding new code to it.
But it's been annoying me more and more, so now that I've decided C99
bool is an acceptable thing to require from our toolchain in the first
place, here's a quite thorough trawl through the source doing
'boolification'. Many variables and function parameters are now typed
as bool rather than int; many assignments of 0 or 1 to those variables
are now spelled 'true' or 'false'.
I managed this thorough conversion with the help of a custom clang
plugin that I wrote to trawl the AST and apply heuristics to point out
where things might want changing. So I've even managed to do a decent
job on parts of the code I haven't looked at in years!
To make the plugin's work easier, I pushed platform front ends
generally in the direction of using standard 'bool' in preference to
platform-specific boolean types like Windows BOOL or GTK's gboolean;
I've left the platform booleans in places they _have_ to be for the
platform APIs to work right, but variables only used by my own code
have been converted wherever I found them.
In a few places there are int values that look very like booleans in
_most_ of the places they're used, but have a rarely-used third value,
or a distinction between different nonzero values that most users
don't care about. In these cases, I've _removed_ uses of 'true' and
'false' for the return values, to emphasise that there's something
more subtle going on than a simple boolean answer:
- the 'multisel' field in dialog.h's list box structure, for which
the GTK front end in particular recognises a difference between 1
and 2 but nearly everything else treats as boolean
- the 'urgent' parameter to plug_receive, where 1 vs 2 tells you
something about the specific location of the urgent pointer, but
most clients only care about 0 vs 'something nonzero'
- the return value of wc_match, where -1 indicates a syntax error in
the wildcard.
- the return values from SSH-1 RSA-key loading functions, which use
-1 for 'wrong passphrase' and 0 for all other failures (so any
caller which already knows it's not loading an _encrypted private_
key can treat them as boolean)
- term->esc_query, and the 'query' parameter in toggle_mode in
terminal.c, which _usually_ hold 0 for ESC[123h or 1 for ESC[?123h,
but can also hold -1 for some other intervening character that we
don't support.
In a few places there's an integer that I haven't turned into a bool
even though it really _can_ only take values 0 or 1 (and, as above,
tried to make the call sites consistent in not calling those values
true and false), on the grounds that I thought it would make it more
confusing to imply that the 0 value was in some sense 'negative' or
bad and the 1 positive or good:
- the return value of plug_accepting uses the POSIXish convention of
0=success and nonzero=error; I think if I made it bool then I'd
also want to reverse its sense, and that's a job for a separate
piece of work.
- the 'screen' parameter to lineptr() in terminal.c, where 0 and 1
represent the default and alternate screens. There's no obvious
reason why one of those should be considered 'true' or 'positive'
or 'success' - they're just indices - so I've left it as int.
ssh_scp_recv had particularly confusing semantics for its previous int
return value: its call sites used '<= 0' to check for error, but it
never actually returned a negative number, just 0 or 1. Now the
function and its call sites agree that it's a bool.
In a couple of places I've renamed variables called 'ret', because I
don't like that name any more - it's unclear whether it means the
return value (in preparation) for the _containing_ function or the
return value received from a subroutine call, and occasionally I've
accidentally used the same variable for both and introduced a bug. So
where one of those got in my way, I've renamed it to 'toret' or 'retd'
(the latter short for 'returned') in line with my usual modern
practice, but I haven't done a thorough job of finding all of them.
Finally, one amusing side effect of doing this is that I've had to
separate quite a few chained assignments. It used to be perfectly fine
to write 'a = b = c = TRUE' when a,b,c were int and TRUE was just a
the 'true' defined by stdbool.h, that idiom provokes a warning from
gcc: 'suggest parentheses around assignment used as truth value'!
2018-11-02 19:23:19 +00:00
|
|
|
bool create_directory(const char *name);
|
2003-09-02 09:00:35 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Concatenate a directory name and a file name. The way this is
|
|
|
|
* done will depend on the OS.
|
|
|
|
*/
|
2015-05-15 10:15:42 +00:00
|
|
|
char *dir_file_cat(const char *dir, const char *file);
|
2003-09-02 09:00:35 +00:00
|
|
|
|
2015-09-24 16:47:10 +00:00
|
|
|
/*
|
|
|
|
* Return a pointer to the portion of str that comes after the last
|
|
|
|
* path component separator.
|
|
|
|
*
|
|
|
|
* If 'local' is false, path component separators are taken to just be
|
|
|
|
* '/', on the assumption that we're discussing the path syntax on the
|
|
|
|
* server. But if 'local' is true, the separators are whatever the
|
|
|
|
* local OS will treat that way - so that includes '\' and ':' on
|
|
|
|
* Windows.
|
|
|
|
*
|
|
|
|
* This function has the annoying strstr() property of taking a const
|
|
|
|
* char * and returning a char *. You should treat it as if it was a
|
|
|
|
* pair of overloaded functions, one mapping mutable->mutable and the
|
|
|
|
* other const->const :-(
|
|
|
|
*/
|
Convert a lot of 'int' variables to 'bool'.
My normal habit these days, in new code, is to treat int and bool as
_almost_ completely separate types. I'm still willing to use C's
implicit test for zero on an integer (e.g. 'if (!blob.len)' is fine,
no need to spell it out as blob.len != 0), but generally, if a
variable is going to be conceptually a boolean, I like to declare it
bool and assign to it using 'true' or 'false' rather than 0 or 1.
PuTTY is an exception, because it predates the C99 bool, and I've
stuck to its existing coding style even when adding new code to it.
But it's been annoying me more and more, so now that I've decided C99
bool is an acceptable thing to require from our toolchain in the first
place, here's a quite thorough trawl through the source doing
'boolification'. Many variables and function parameters are now typed
as bool rather than int; many assignments of 0 or 1 to those variables
are now spelled 'true' or 'false'.
I managed this thorough conversion with the help of a custom clang
plugin that I wrote to trawl the AST and apply heuristics to point out
where things might want changing. So I've even managed to do a decent
job on parts of the code I haven't looked at in years!
To make the plugin's work easier, I pushed platform front ends
generally in the direction of using standard 'bool' in preference to
platform-specific boolean types like Windows BOOL or GTK's gboolean;
I've left the platform booleans in places they _have_ to be for the
platform APIs to work right, but variables only used by my own code
have been converted wherever I found them.
In a few places there are int values that look very like booleans in
_most_ of the places they're used, but have a rarely-used third value,
or a distinction between different nonzero values that most users
don't care about. In these cases, I've _removed_ uses of 'true' and
'false' for the return values, to emphasise that there's something
more subtle going on than a simple boolean answer:
- the 'multisel' field in dialog.h's list box structure, for which
the GTK front end in particular recognises a difference between 1
and 2 but nearly everything else treats as boolean
- the 'urgent' parameter to plug_receive, where 1 vs 2 tells you
something about the specific location of the urgent pointer, but
most clients only care about 0 vs 'something nonzero'
- the return value of wc_match, where -1 indicates a syntax error in
the wildcard.
- the return values from SSH-1 RSA-key loading functions, which use
-1 for 'wrong passphrase' and 0 for all other failures (so any
caller which already knows it's not loading an _encrypted private_
key can treat them as boolean)
- term->esc_query, and the 'query' parameter in toggle_mode in
terminal.c, which _usually_ hold 0 for ESC[123h or 1 for ESC[?123h,
but can also hold -1 for some other intervening character that we
don't support.
In a few places there's an integer that I haven't turned into a bool
even though it really _can_ only take values 0 or 1 (and, as above,
tried to make the call sites consistent in not calling those values
true and false), on the grounds that I thought it would make it more
confusing to imply that the 0 value was in some sense 'negative' or
bad and the 1 positive or good:
- the return value of plug_accepting uses the POSIXish convention of
0=success and nonzero=error; I think if I made it bool then I'd
also want to reverse its sense, and that's a job for a separate
piece of work.
- the 'screen' parameter to lineptr() in terminal.c, where 0 and 1
represent the default and alternate screens. There's no obvious
reason why one of those should be considered 'true' or 'positive'
or 'success' - they're just indices - so I've left it as int.
ssh_scp_recv had particularly confusing semantics for its previous int
return value: its call sites used '<= 0' to check for error, but it
never actually returned a negative number, just 0 or 1. Now the
function and its call sites agree that it's a bool.
In a couple of places I've renamed variables called 'ret', because I
don't like that name any more - it's unclear whether it means the
return value (in preparation) for the _containing_ function or the
return value received from a subroutine call, and occasionally I've
accidentally used the same variable for both and introduced a bug. So
where one of those got in my way, I've renamed it to 'toret' or 'retd'
(the latter short for 'returned') in line with my usual modern
practice, but I haven't done a thorough job of finding all of them.
Finally, one amusing side effect of doing this is that I've had to
separate quite a few chained assignments. It used to be perfectly fine
to write 'a = b = c = TRUE' when a,b,c were int and TRUE was just a
the 'true' defined by stdbool.h, that idiom provokes a warning from
gcc: 'suggest parentheses around assignment used as truth value'!
2018-11-02 19:23:19 +00:00
|
|
|
char *stripslashes(const char *str, bool local);
|
2015-09-24 16:47:10 +00:00
|
|
|
|
2003-09-02 09:00:35 +00:00
|
|
|
#endif /* PUTTY_PSFTP_H */
|