Blame src/misc.c

Packit 6c0a39
/*
Packit 6c0a39
 * misc.c - useful client functions
Packit 6c0a39
 *
Packit 6c0a39
 * This file is part of the SSH Library
Packit 6c0a39
 *
Packit 6c0a39
 * Copyright (c) 2003-2009 by Aris Adamantiadis
Packit 6c0a39
 * Copyright (c) 2008-2009 by Andreas Schneider <asn@cryptomilk.org>
Packit 6c0a39
 *
Packit 6c0a39
 * The SSH Library is free software; you can redistribute it and/or modify
Packit 6c0a39
 * it under the terms of the GNU Lesser General Public License as published by
Packit 6c0a39
 * the Free Software Foundation; either version 2.1 of the License, or (at your
Packit 6c0a39
 * option) any later version.
Packit 6c0a39
 *
Packit 6c0a39
 * The SSH Library is distributed in the hope that it will be useful, but
Packit 6c0a39
 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
Packit 6c0a39
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
Packit 6c0a39
 * License for more details.
Packit 6c0a39
 *
Packit 6c0a39
 * You should have received a copy of the GNU Lesser General Public License
Packit 6c0a39
 * along with the SSH Library; see the file COPYING.  If not, write to
Packit 6c0a39
 * the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
Packit 6c0a39
 * MA 02111-1307, USA.
Packit 6c0a39
 */
Packit 6c0a39
Packit 6c0a39
#include "config.h"
Packit 6c0a39
Packit 6c0a39
#ifndef _WIN32
Packit 6c0a39
/* This is needed for a standard getpwuid_r on opensolaris */
Packit 6c0a39
#define _POSIX_PTHREAD_SEMANTICS
Packit 6c0a39
#include <pwd.h>
Packit 6c0a39
#include <sys/types.h>
Packit 6c0a39
#include <sys/socket.h>
Packit 6c0a39
#include <netinet/in.h>
Packit 6c0a39
#include <arpa/inet.h>
Packit 6c0a39
Packit 6c0a39
#endif /* _WIN32 */
Packit 6c0a39
Packit 6c0a39
#include <errno.h>
Packit 6c0a39
#include <limits.h>
Packit 6c0a39
#include <stdio.h>
Packit 6c0a39
#include <string.h>
Packit 6c0a39
#include <stdlib.h>
Packit 6c0a39
#include <sys/stat.h>
Packit 6c0a39
#include <sys/types.h>
Packit 6c0a39
#include <ctype.h>
Packit 6c0a39
#include <time.h>
Packit 6c0a39
#ifdef HAVE_SYS_TIME_H
Packit 6c0a39
#include <sys/time.h>
Packit 6c0a39
#endif /* HAVE_SYS_TIME_H */
Packit 6c0a39
Packit 6c0a39
Packit 6c0a39
#ifdef _WIN32
Packit 6c0a39
Packit 6c0a39
#ifndef _WIN32_IE
Packit 6c0a39
# define _WIN32_IE 0x0501 // SHGetSpecialFolderPath
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
#include <winsock2.h> // Must be the first to include
Packit 6c0a39
#include <ws2tcpip.h>
Packit 6c0a39
#include <shlobj.h>
Packit 6c0a39
#include <direct.h>
Packit 6c0a39
Packit 6c0a39
#ifdef HAVE_IO_H
Packit 6c0a39
#include <io.h>
Packit 6c0a39
#endif /* HAVE_IO_H */
Packit 6c0a39
Packit 6c0a39
#endif /* _WIN32 */
Packit 6c0a39
Packit 6c0a39
#include "libssh/priv.h"
Packit 6c0a39
#include "libssh/misc.h"
Packit 6c0a39
#include "libssh/session.h"
Packit 6c0a39
Packit 6c0a39
#ifdef HAVE_LIBGCRYPT
Packit 6c0a39
#define GCRYPT_STRING "/gnutls"
Packit 6c0a39
#else
Packit 6c0a39
#define GCRYPT_STRING ""
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
#ifdef HAVE_LIBCRYPTO
Packit 6c0a39
#define CRYPTO_STRING "/openssl"
Packit 6c0a39
#else
Packit 6c0a39
#define CRYPTO_STRING ""
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
#ifdef HAVE_LIBMBEDCRYPTO
Packit 6c0a39
#define MBED_STRING "/mbedtls"
Packit 6c0a39
#else
Packit 6c0a39
#define MBED_STRING ""
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
#ifdef WITH_ZLIB
Packit 6c0a39
#define ZLIB_STRING "/zlib"
Packit 6c0a39
#else
Packit 6c0a39
#define ZLIB_STRING ""
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @defgroup libssh_misc The SSH helper functions.
Packit 6c0a39
 * @ingroup libssh
Packit 6c0a39
 *
Packit 6c0a39
 * Different helper functions used in the SSH Library.
Packit 6c0a39
 *
Packit 6c0a39
 * @{
Packit 6c0a39
 */
Packit 6c0a39
Packit 6c0a39
#ifdef _WIN32
Packit 6c0a39
char *ssh_get_user_home_dir(void) {
Packit 6c0a39
  char tmp[MAX_PATH] = {0};
Packit 6c0a39
  char *szPath = NULL;
Packit 6c0a39
Packit 6c0a39
  if (SHGetSpecialFolderPathA(NULL, tmp, CSIDL_PROFILE, TRUE)) {
Packit 6c0a39
    szPath = malloc(strlen(tmp) + 1);
Packit 6c0a39
    if (szPath == NULL) {
Packit 6c0a39
      return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    strcpy(szPath, tmp);
Packit 6c0a39
    return szPath;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return NULL;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/* we have read access on file */
Packit 6c0a39
int ssh_file_readaccess_ok(const char *file) {
Packit 6c0a39
  if (_access(file, 4) < 0) {
Packit 6c0a39
    return 0;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return 1;
Packit 6c0a39
}
Packit 6c0a39
Packit Service fcc0d2
/**
Packit Service fcc0d2
 * @brief Check if the given path is an existing directory and that is
Packit Service fcc0d2
 * accessible for writing.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in] path Path to the directory to be checked
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @return Return 1 if the directory exists and is accessible; 0 otherwise
Packit Service fcc0d2
 * */
Packit Service fcc0d2
int ssh_dir_writeable(const char *path)
Packit Service fcc0d2
{
Packit Service fcc0d2
    struct _stat buffer;
Packit Service fcc0d2
    int rc;
Packit Service fcc0d2
Packit Service fcc0d2
    rc = _stat(path, &buffer);
Packit Service fcc0d2
    if (rc < 0) {
Packit Service fcc0d2
        return 0;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    if ((buffer.st_mode & _S_IFDIR) && (buffer.st_mode & _S_IWRITE)) {
Packit Service fcc0d2
        return 1;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    return 0;
Packit Service fcc0d2
}
Packit Service fcc0d2
Packit 6c0a39
#define SSH_USEC_IN_SEC         1000000LL
Packit 6c0a39
#define SSH_SECONDS_SINCE_1601  11644473600LL
Packit 6c0a39
Packit 6c0a39
int gettimeofday(struct timeval *__p, void *__t) {
Packit 6c0a39
  union {
Packit 6c0a39
    unsigned long long ns100; /* time since 1 Jan 1601 in 100ns units */
Packit 6c0a39
    FILETIME ft;
Packit 6c0a39
  } now;
Packit 6c0a39
Packit 6c0a39
  GetSystemTimeAsFileTime (&now.ft);
Packit 6c0a39
  __p->tv_usec = (long) ((now.ns100 / 10LL) % SSH_USEC_IN_SEC);
Packit 6c0a39
  __p->tv_sec  = (long)(((now.ns100 / 10LL ) / SSH_USEC_IN_SEC) - SSH_SECONDS_SINCE_1601);
Packit 6c0a39
Packit 6c0a39
  return (0);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
char *ssh_get_local_username(void) {
Packit 6c0a39
    DWORD size = 0;
Packit 6c0a39
    char *user;
Packit 6c0a39
Packit 6c0a39
    /* get the size */
Packit 6c0a39
    GetUserName(NULL, &size);
Packit 6c0a39
Packit 6c0a39
    user = (char *) malloc(size);
Packit 6c0a39
    if (user == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    if (GetUserName(user, &size)) {
Packit 6c0a39
        return user;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return NULL;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int ssh_is_ipaddr_v4(const char *str) {
Packit 6c0a39
    struct sockaddr_storage ss;
Packit 6c0a39
    int sslen = sizeof(ss);
Packit 6c0a39
    int rc = SOCKET_ERROR;
Packit 6c0a39
Packit 6c0a39
    /* WSAStringToAddressA thinks that 0.0.0 is a valid IP */
Packit 6c0a39
    if (strlen(str) < 7) {
Packit 6c0a39
        return 0;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    rc = WSAStringToAddressA((LPSTR) str,
Packit 6c0a39
                             AF_INET,
Packit 6c0a39
                             NULL,
Packit 6c0a39
                             (struct sockaddr*)&ss,
Packit 6c0a39
                             &sslen);
Packit 6c0a39
    if (rc == 0) {
Packit 6c0a39
        return 1;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int ssh_is_ipaddr(const char *str) {
Packit 6c0a39
    int rc = SOCKET_ERROR;
Packit 6c0a39
Packit 6c0a39
    if (strchr(str, ':')) {
Packit 6c0a39
        struct sockaddr_storage ss;
Packit 6c0a39
        int sslen = sizeof(ss);
Packit 6c0a39
Packit 6c0a39
        /* TODO link-local (IP:v6:addr%ifname). */
Packit 6c0a39
        rc = WSAStringToAddressA((LPSTR) str,
Packit 6c0a39
                                 AF_INET6,
Packit 6c0a39
                                 NULL,
Packit 6c0a39
                                 (struct sockaddr*)&ss,
Packit 6c0a39
                                 &sslen);
Packit 6c0a39
        if (rc == 0) {
Packit 6c0a39
            return 1;
Packit 6c0a39
        }
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return ssh_is_ipaddr_v4(str);
Packit 6c0a39
}
Packit 6c0a39
#else /* _WIN32 */
Packit 6c0a39
Packit 6c0a39
#ifndef NSS_BUFLEN_PASSWD
Packit 6c0a39
#define NSS_BUFLEN_PASSWD 4096
Packit 6c0a39
#endif /* NSS_BUFLEN_PASSWD */
Packit 6c0a39
Packit 6c0a39
char *ssh_get_user_home_dir(void)
Packit 6c0a39
{
Packit 6c0a39
    char *szPath = NULL;
Packit 6c0a39
    struct passwd pwd;
Packit 6c0a39
    struct passwd *pwdbuf = NULL;
Packit 6c0a39
    char buf[NSS_BUFLEN_PASSWD] = {0};
Packit 6c0a39
    int rc;
Packit 6c0a39
Packit 6c0a39
    rc = getpwuid_r(getuid(), &pwd, buf, NSS_BUFLEN_PASSWD, &pwdbuf);
Packit 6c0a39
    if (rc != 0 || pwdbuf == NULL ) {
Packit 6c0a39
        szPath = getenv("HOME");
Packit 6c0a39
        if (szPath == NULL) {
Packit 6c0a39
            return NULL;
Packit 6c0a39
        }
Packit 6c0a39
        snprintf(buf, sizeof(buf), "%s", szPath);
Packit 6c0a39
Packit 6c0a39
        return strdup(buf);
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    szPath = strdup(pwd.pw_dir);
Packit 6c0a39
Packit 6c0a39
    return szPath;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/* we have read access on file */
Packit 6c0a39
int ssh_file_readaccess_ok(const char *file)
Packit 6c0a39
{
Packit 6c0a39
    if (access(file, R_OK) < 0) {
Packit 6c0a39
        return 0;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return 1;
Packit 6c0a39
}
Packit 6c0a39
Packit Service fcc0d2
/**
Packit Service fcc0d2
 * @brief Check if the given path is an existing directory and that is
Packit Service fcc0d2
 * accessible for writing.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in] path Path to the directory to be checked
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @return Return 1 if the directory exists and is accessible; 0 otherwise
Packit Service fcc0d2
 * */
Packit Service fcc0d2
int ssh_dir_writeable(const char *path)
Packit Service fcc0d2
{
Packit Service fcc0d2
    struct stat buffer;
Packit Service fcc0d2
    int rc;
Packit Service fcc0d2
Packit Service fcc0d2
    rc = stat(path, &buffer);
Packit Service fcc0d2
    if (rc < 0) {
Packit Service fcc0d2
        return 0;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    if (S_ISDIR(buffer.st_mode) && (buffer.st_mode & S_IWRITE)) {
Packit Service fcc0d2
        return 1;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    return 0;
Packit Service fcc0d2
}
Packit Service fcc0d2
Packit 6c0a39
char *ssh_get_local_username(void)
Packit 6c0a39
{
Packit 6c0a39
    struct passwd pwd;
Packit 6c0a39
    struct passwd *pwdbuf = NULL;
Packit 6c0a39
    char buf[NSS_BUFLEN_PASSWD];
Packit 6c0a39
    char *name;
Packit 6c0a39
    int rc;
Packit 6c0a39
Packit 6c0a39
    rc = getpwuid_r(getuid(), &pwd, buf, NSS_BUFLEN_PASSWD, &pwdbuf);
Packit 6c0a39
    if (rc != 0 || pwdbuf == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    name = strdup(pwd.pw_name);
Packit 6c0a39
Packit 6c0a39
    if (name == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return name;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int ssh_is_ipaddr_v4(const char *str) {
Packit 6c0a39
    int rc = -1;
Packit 6c0a39
    struct in_addr dest;
Packit 6c0a39
Packit 6c0a39
    rc = inet_pton(AF_INET, str, &dest);
Packit 6c0a39
    if (rc > 0) {
Packit 6c0a39
        return 1;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int ssh_is_ipaddr(const char *str) {
Packit 6c0a39
    int rc = -1;
Packit 6c0a39
Packit 6c0a39
    if (strchr(str, ':')) {
Packit 6c0a39
        struct in6_addr dest6;
Packit 6c0a39
Packit 6c0a39
        /* TODO link-local (IP:v6:addr%ifname). */
Packit 6c0a39
        rc = inet_pton(AF_INET6, str, &dest6);
Packit 6c0a39
        if (rc > 0) {
Packit 6c0a39
            return 1;
Packit 6c0a39
        }
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    return ssh_is_ipaddr_v4(str);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
#endif /* _WIN32 */
Packit 6c0a39
Packit 6c0a39
char *ssh_lowercase(const char* str) {
Packit 6c0a39
  char *new, *p;
Packit 6c0a39
Packit 6c0a39
  if (str == NULL) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  new = strdup(str);
Packit 6c0a39
  if (new == NULL) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  for (p = new; *p; p++) {
Packit 6c0a39
    *p = tolower(*p);
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return new;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
char *ssh_hostport(const char *host, int port)
Packit 6c0a39
{
Packit 6c0a39
    char *dest = NULL;
Packit 6c0a39
    size_t len;
Packit 6c0a39
Packit 6c0a39
    if (host == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    /* 3 for []:, 5 for 65536 and 1 for nul */
Packit 6c0a39
    len = strlen(host) + 3 + 5 + 1;
Packit 6c0a39
    dest = malloc(len);
Packit 6c0a39
    if (dest == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
    snprintf(dest, len, "[%s]:%d", host, port);
Packit 6c0a39
Packit 6c0a39
    return dest;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief Convert a buffer into a colon separated hex string.
Packit 6c0a39
 * The caller has to free the memory.
Packit 6c0a39
 *
Packit 6c0a39
 * @param  what         What should be converted to a hex string.
Packit 6c0a39
 *
Packit 6c0a39
 * @param  len          Length of the buffer to convert.
Packit 6c0a39
 *
Packit 6c0a39
 * @return              The hex string or NULL on error.
Packit 6c0a39
 *
Packit 6c0a39
 * @see ssh_string_free_char()
Packit 6c0a39
 */
Packit 6c0a39
char *ssh_get_hexa(const unsigned char *what, size_t len) {
Packit 6c0a39
    const char h[] = "0123456789abcdef";
Packit 6c0a39
    char *hexa;
Packit 6c0a39
    size_t i;
Packit 6c0a39
    size_t hlen = len * 3;
Packit 6c0a39
Packit 6c0a39
    if (len > (UINT_MAX - 1) / 3) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    hexa = malloc(hlen + 1);
Packit 6c0a39
    if (hexa == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    for (i = 0; i < len; i++) {
Packit 6c0a39
        hexa[i * 3] = h[(what[i] >> 4) & 0xF];
Packit 6c0a39
        hexa[i * 3 + 1] = h[what[i] & 0xF];
Packit 6c0a39
        hexa[i * 3 + 2] = ':';
Packit 6c0a39
    }
Packit 6c0a39
    hexa[hlen - 1] = '\0';
Packit 6c0a39
Packit 6c0a39
    return hexa;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @deprecated          Please use ssh_print_hash() instead
Packit 6c0a39
 */
Packit 6c0a39
void ssh_print_hexa(const char *descr, const unsigned char *what, size_t len) {
Packit 6c0a39
    char *hexa = ssh_get_hexa(what, len);
Packit 6c0a39
Packit 6c0a39
    if (hexa == NULL) {
Packit 6c0a39
      return;
Packit 6c0a39
    }
Packit 6c0a39
    fprintf(stderr, "%s: %s\n", descr, hexa);
Packit 6c0a39
Packit 6c0a39
    free(hexa);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit Service fcc0d2
 * @brief Log the content of a buffer in hexadecimal format, similar to the
Packit Service fcc0d2
 * output of 'hexdump -C' command.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * The first logged line is the given description followed by the length.
Packit Service fcc0d2
 * Then the content of the buffer is logged 16 bytes per line in the following
Packit Service fcc0d2
 * format:
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * (offset) (first 8 bytes) (last 8 bytes) (the 16 bytes as ASCII char values)
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * The output for a 16 bytes array containing values from 0x00 to 0x0f would be:
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * "Example (16 bytes):"
Packit Service fcc0d2
 * "  00000000  00 01 02 03 04 05 06 07  08 09 0a 0b 0c 0d 0e 0f  ................"
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * The value for each byte as corresponding ASCII character is printed at the
Packit Service fcc0d2
 * end if the value is printable. Otherwise it is replace with '.'.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in] descr A description for the content to be logged
Packit Service fcc0d2
 * @param[in] what  The buffer to be logged
Packit Service fcc0d2
 * @param[in] len   The length of the buffer given in what
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @note If a too long description is provided (which would result in a first
Packit Service fcc0d2
 * line longer than 80 bytes), the function will fail.
Packit Service fcc0d2
 */
Packit Service fcc0d2
void ssh_log_hexdump(const char *descr, const unsigned char *what, size_t len)
Packit Service fcc0d2
{
Packit Service fcc0d2
    size_t i;
Packit Service fcc0d2
    char ascii[17];
Packit Service fcc0d2
    const unsigned char *pc = NULL;
Packit Service fcc0d2
    size_t count = 0;
Packit Service fcc0d2
    ssize_t printed = 0;
Packit Service fcc0d2
Packit Service fcc0d2
    /* The required buffer size is calculated from:
Packit Service fcc0d2
     *
Packit Service fcc0d2
     *  2 bytes for spaces at the beginning
Packit Service fcc0d2
     *  8 bytes for the offset
Packit Service fcc0d2
     *  2 bytes for spaces
Packit Service fcc0d2
     * 24 bytes to print the first 8 bytes + spaces
Packit Service fcc0d2
     *  1 byte for an extra space
Packit Service fcc0d2
     * 24 bytes to print next 8 bytes + spaces
Packit Service fcc0d2
     *  2 bytes for extra spaces
Packit Service fcc0d2
     * 16 bytes for the content as ASCII characters at the end
Packit Service fcc0d2
     *  1 byte for the ending '\0'
Packit Service fcc0d2
     *
Packit Service fcc0d2
     * Resulting in 80 bytes.
Packit Service fcc0d2
     *
Packit Service fcc0d2
     * Except for the first line (description + size), all lines have fixed
Packit Service fcc0d2
     * length. If a too long description is used, the function will fail.
Packit Service fcc0d2
     * */
Packit Service fcc0d2
    char buffer[80];
Packit Service fcc0d2
Packit Service fcc0d2
    /* Print description */
Packit Service fcc0d2
    if (descr != NULL) {
Packit Service fcc0d2
        printed = snprintf(buffer, sizeof(buffer), "%s ", descr);
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        count += printed;
Packit Service fcc0d2
    } else {
Packit Service fcc0d2
        printed = snprintf(buffer, sizeof(buffer), "(NULL description) ");
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        count += printed;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    if (len == 0) {
Packit Service fcc0d2
        printed = snprintf(buffer + count, sizeof(buffer) - count,
Packit Service fcc0d2
                           "(zero length):");
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        SSH_LOG(SSH_LOG_DEBUG, "%s", buffer);
Packit Service fcc0d2
        return;
Packit Service fcc0d2
    } else {
Packit Service fcc0d2
        printed = snprintf(buffer + count, sizeof(buffer) - count,
Packit Service fcc0d2
                           "(%zu bytes):", len);
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        count += printed;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    if (what == NULL) {
Packit Service fcc0d2
        printed = snprintf(buffer + count, sizeof(buffer) - count,
Packit Service fcc0d2
                           "(NULL)");
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        SSH_LOG(SSH_LOG_DEBUG, "%s", buffer);
Packit Service fcc0d2
        return;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    SSH_LOG(SSH_LOG_DEBUG, "%s", buffer);
Packit Service fcc0d2
Packit Service fcc0d2
    /* Reset state */
Packit Service fcc0d2
    count = 0;
Packit Service fcc0d2
    pc = what;
Packit Service fcc0d2
Packit Service fcc0d2
    for (i = 0; i < len; i++) {
Packit Service fcc0d2
        /* Add one space after printing 8 bytes */
Packit Service fcc0d2
        if ((i % 8) == 0) {
Packit Service fcc0d2
            if (i != 0) {
Packit Service fcc0d2
                printed = snprintf(buffer + count, sizeof(buffer) - count, " ");
Packit Service fcc0d2
                if (printed < 0) {
Packit Service fcc0d2
                    goto error;
Packit Service fcc0d2
                }
Packit Service fcc0d2
                count += printed;
Packit Service fcc0d2
            }
Packit Service fcc0d2
        }
Packit Service fcc0d2
Packit Service fcc0d2
        /* Log previous line and reset state for new line */
Packit Service fcc0d2
        if ((i % 16) == 0) {
Packit Service fcc0d2
            if (i != 0) {
Packit Service fcc0d2
                printed = snprintf(buffer + count, sizeof(buffer) - count,
Packit Service fcc0d2
                                   "  %s", ascii);
Packit Service fcc0d2
                if (printed < 0) {
Packit Service fcc0d2
                    goto error;
Packit Service fcc0d2
                }
Packit Service fcc0d2
                SSH_LOG(SSH_LOG_DEBUG, "%s", buffer);
Packit Service fcc0d2
                count = 0;
Packit Service fcc0d2
            }
Packit Service fcc0d2
Packit Service fcc0d2
            /* Start a new line with the offset */
Packit Service fcc0d2
            printed = snprintf(buffer, sizeof(buffer),
Packit Service fcc0d2
                               "  %08zx ", i);
Packit Service fcc0d2
            if (printed < 0) {
Packit Service fcc0d2
                goto error;
Packit Service fcc0d2
            }
Packit Service fcc0d2
            count += printed;
Packit Service fcc0d2
        }
Packit Service fcc0d2
Packit Service fcc0d2
        /* Print the current byte hexadecimal representation */
Packit Service fcc0d2
        printed = snprintf(buffer + count, sizeof(buffer) - count,
Packit Service fcc0d2
                           " %02x", pc[i]);
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        count += printed;
Packit Service fcc0d2
Packit Service fcc0d2
        /* If printable, store the ASCII character */
Packit Service fcc0d2
        if (isprint(pc[i])) {
Packit Service fcc0d2
            ascii[i % 16] = pc[i];
Packit Service fcc0d2
        } else {
Packit Service fcc0d2
            ascii[i % 16] = '.';
Packit Service fcc0d2
        }
Packit Service fcc0d2
        ascii[(i % 16) + 1] = '\0';
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    /* Add padding if not exactly 16 characters */
Packit Service fcc0d2
    while ((i % 16) != 0) {
Packit Service fcc0d2
        /* Add one space after printing 8 bytes */
Packit Service fcc0d2
        if ((i % 8) == 0) {
Packit Service fcc0d2
            if (i != 0) {
Packit Service fcc0d2
                printed = snprintf(buffer + count, sizeof(buffer) - count, " ");
Packit Service fcc0d2
                if (printed < 0) {
Packit Service fcc0d2
                    goto error;
Packit Service fcc0d2
                }
Packit Service fcc0d2
                count += printed;
Packit Service fcc0d2
            }
Packit Service fcc0d2
        }
Packit Service fcc0d2
Packit Service fcc0d2
        printed = snprintf(buffer + count, sizeof(buffer) - count, "   ");
Packit Service fcc0d2
        if (printed < 0) {
Packit Service fcc0d2
            goto error;
Packit Service fcc0d2
        }
Packit Service fcc0d2
        count += printed;
Packit Service fcc0d2
        i++;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    /* Print the last printable part */
Packit Service fcc0d2
    printed = snprintf(buffer + count, sizeof(buffer) - count,
Packit Service fcc0d2
                       "   %s", ascii);
Packit Service fcc0d2
    if (printed < 0) {
Packit Service fcc0d2
        goto error;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    SSH_LOG(SSH_LOG_DEBUG, "%s", buffer);
Packit Service fcc0d2
Packit Service fcc0d2
    return;
Packit Service fcc0d2
Packit Service fcc0d2
error:
Packit Service fcc0d2
    SSH_LOG(SSH_LOG_WARN, "Could not print to buffer");
Packit Service fcc0d2
    return;
Packit Service fcc0d2
}
Packit Service fcc0d2
Packit Service fcc0d2
/**
Packit 6c0a39
 * @brief Check if libssh is the required version or get the version
Packit 6c0a39
 * string.
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  req_version The version required.
Packit 6c0a39
 *
Packit 6c0a39
 * @return              If the version of libssh is newer than the version
Packit 6c0a39
 *                      required it will return a version string.
Packit 6c0a39
 *                      NULL if the version is older.
Packit 6c0a39
 *
Packit 6c0a39
 * Example:
Packit 6c0a39
 *
Packit 6c0a39
 * @code
Packit 6c0a39
 *  if (ssh_version(SSH_VERSION_INT(0,2,1)) == NULL) {
Packit 6c0a39
 *    fprintf(stderr, "libssh version is too old!\n");
Packit 6c0a39
 *    exit(1);
Packit 6c0a39
 *  }
Packit 6c0a39
 *
Packit 6c0a39
 *  if (debug) {
Packit 6c0a39
 *    printf("libssh %s\n", ssh_version(0));
Packit 6c0a39
 *  }
Packit 6c0a39
 * @endcode
Packit 6c0a39
 */
Packit 6c0a39
const char *ssh_version(int req_version) {
Packit 6c0a39
  if (req_version <= LIBSSH_VERSION_INT) {
Packit 6c0a39
    return SSH_STRINGIFY(LIBSSH_VERSION) GCRYPT_STRING CRYPTO_STRING MBED_STRING
Packit 6c0a39
      ZLIB_STRING;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return NULL;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
struct ssh_list *ssh_list_new(void) {
Packit 6c0a39
  struct ssh_list *ret=malloc(sizeof(struct ssh_list));
Packit 6c0a39
  if(!ret)
Packit 6c0a39
    return NULL;
Packit 6c0a39
  ret->root=ret->end=NULL;
Packit 6c0a39
  return ret;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
void ssh_list_free(struct ssh_list *list){
Packit 6c0a39
  struct ssh_iterator *ptr,*next;
Packit 6c0a39
  if(!list)
Packit 6c0a39
    return;
Packit 6c0a39
  ptr=list->root;
Packit 6c0a39
  while(ptr){
Packit 6c0a39
    next=ptr->next;
Packit 6c0a39
    SAFE_FREE(ptr);
Packit 6c0a39
    ptr=next;
Packit 6c0a39
  }
Packit 6c0a39
  SAFE_FREE(list);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
struct ssh_iterator *ssh_list_get_iterator(const struct ssh_list *list){
Packit 6c0a39
  if(!list)
Packit 6c0a39
    return NULL;
Packit 6c0a39
  return list->root;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
struct ssh_iterator *ssh_list_find(const struct ssh_list *list, void *value){
Packit 6c0a39
  struct ssh_iterator *it;
Packit 6c0a39
  for(it = ssh_list_get_iterator(list); it != NULL ;it=it->next)
Packit 6c0a39
    if(it->data==value)
Packit 6c0a39
      return it;
Packit 6c0a39
  return NULL;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief Get the number of elements in the list
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  list     The list to count.
Packit 6c0a39
 *
Packit 6c0a39
 * @return The number of elements in the list.
Packit 6c0a39
 */
Packit 6c0a39
size_t ssh_list_count(const struct ssh_list *list)
Packit 6c0a39
{
Packit 6c0a39
  struct ssh_iterator *it = NULL;
Packit 6c0a39
  int count = 0;
Packit 6c0a39
Packit 6c0a39
  for (it = ssh_list_get_iterator(list); it != NULL ; it = it->next) {
Packit 6c0a39
      count++;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return count;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
static struct ssh_iterator *ssh_iterator_new(const void *data){
Packit 6c0a39
  struct ssh_iterator *iterator=malloc(sizeof(struct ssh_iterator));
Packit 6c0a39
  if(!iterator)
Packit 6c0a39
    return NULL;
Packit 6c0a39
  iterator->next=NULL;
Packit 6c0a39
  iterator->data=data;
Packit 6c0a39
  return iterator;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int ssh_list_append(struct ssh_list *list,const void *data){
Packit 6c0a39
  struct ssh_iterator *iterator = NULL;
Packit 6c0a39
Packit 6c0a39
  if (list == NULL) {
Packit 6c0a39
      return SSH_ERROR;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  iterator = ssh_iterator_new(data);
Packit 6c0a39
  if (iterator == NULL) {
Packit 6c0a39
      return SSH_ERROR;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if(!list->end){
Packit 6c0a39
    /* list is empty */
Packit 6c0a39
    list->root=list->end=iterator;
Packit 6c0a39
  } else {
Packit 6c0a39
    /* put it on end of list */
Packit 6c0a39
    list->end->next=iterator;
Packit 6c0a39
    list->end=iterator;
Packit 6c0a39
  }
Packit 6c0a39
  return SSH_OK;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int ssh_list_prepend(struct ssh_list *list, const void *data){
Packit 6c0a39
  struct ssh_iterator *it = NULL;
Packit 6c0a39
Packit 6c0a39
  if (list == NULL) {
Packit 6c0a39
      return SSH_ERROR;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  it = ssh_iterator_new(data);
Packit 6c0a39
  if (it == NULL) {
Packit 6c0a39
    return SSH_ERROR;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (list->end == NULL) {
Packit 6c0a39
    /* list is empty */
Packit 6c0a39
    list->root = list->end = it;
Packit 6c0a39
  } else {
Packit 6c0a39
    /* set as new root */
Packit 6c0a39
    it->next = list->root;
Packit 6c0a39
    list->root = it;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return SSH_OK;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
void ssh_list_remove(struct ssh_list *list, struct ssh_iterator *iterator){
Packit 6c0a39
  struct ssh_iterator *ptr,*prev;
Packit 6c0a39
Packit 6c0a39
  if (list == NULL) {
Packit 6c0a39
      return;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  prev=NULL;
Packit 6c0a39
  ptr=list->root;
Packit 6c0a39
  while(ptr && ptr != iterator){
Packit 6c0a39
    prev=ptr;
Packit 6c0a39
    ptr=ptr->next;
Packit 6c0a39
  }
Packit 6c0a39
  if(!ptr){
Packit 6c0a39
    /* we did not find the element */
Packit 6c0a39
    return;
Packit 6c0a39
  }
Packit 6c0a39
  /* unlink it */
Packit 6c0a39
  if(prev)
Packit 6c0a39
    prev->next=ptr->next;
Packit 6c0a39
  /* if iterator was the head */
Packit 6c0a39
  if(list->root == iterator)
Packit 6c0a39
    list->root=iterator->next;
Packit 6c0a39
  /* if iterator was the tail */
Packit 6c0a39
  if(list->end == iterator)
Packit 6c0a39
    list->end = prev;
Packit 6c0a39
  SAFE_FREE(iterator);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @internal
Packit 6c0a39
 *
Packit 6c0a39
 * @brief Removes the top element of the list and returns the data value
Packit 6c0a39
 * attached to it.
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in[  list     The ssh_list to remove the element.
Packit 6c0a39
 *
Packit 6c0a39
 * @returns             A pointer to the element being stored in head, or NULL
Packit 6c0a39
 *                      if the list is empty.
Packit 6c0a39
 */
Packit 6c0a39
const void *_ssh_list_pop_head(struct ssh_list *list){
Packit 6c0a39
  struct ssh_iterator *iterator = NULL;
Packit 6c0a39
  const void *data = NULL;
Packit 6c0a39
Packit 6c0a39
  if (list == NULL) {
Packit 6c0a39
      return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  iterator = list->root;
Packit 6c0a39
  if (iterator == NULL) {
Packit 6c0a39
      return NULL;
Packit 6c0a39
  }
Packit 6c0a39
  data=iterator->data;
Packit 6c0a39
  list->root=iterator->next;
Packit 6c0a39
  if(list->end==iterator)
Packit 6c0a39
    list->end=NULL;
Packit 6c0a39
  SAFE_FREE(iterator);
Packit 6c0a39
  return data;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief Parse directory component.
Packit 6c0a39
 *
Packit 6c0a39
 * dirname breaks a null-terminated pathname string into a directory component.
Packit 6c0a39
 * In the usual case, ssh_dirname() returns the string up to, but not including,
Packit 6c0a39
 * the final '/'. Trailing '/' characters are  not  counted as part of the
Packit 6c0a39
 * pathname. The caller must free the memory.
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  path     The path to parse.
Packit 6c0a39
 *
Packit 6c0a39
 * @return              The dirname of path or NULL if we can't allocate memory.
Packit 6c0a39
 *                      If path does not contain a slash, c_dirname() returns
Packit 6c0a39
 *                      the string ".".  If path is the string "/", it returns
Packit 6c0a39
 *                      the string "/". If path is NULL or an empty string,
Packit 6c0a39
 *                      "." is returned.
Packit 6c0a39
 */
Packit 6c0a39
char *ssh_dirname (const char *path) {
Packit 6c0a39
  char *new = NULL;
Packit 6c0a39
  size_t len;
Packit 6c0a39
Packit 6c0a39
  if (path == NULL || *path == '\0') {
Packit 6c0a39
    return strdup(".");
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  len = strlen(path);
Packit 6c0a39
Packit 6c0a39
  /* Remove trailing slashes */
Packit 6c0a39
  while(len > 0 && path[len - 1] == '/') --len;
Packit 6c0a39
Packit 6c0a39
  /* We have only slashes */
Packit 6c0a39
  if (len == 0) {
Packit 6c0a39
    return strdup("/");
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  /* goto next slash */
Packit 6c0a39
  while(len > 0 && path[len - 1] != '/') --len;
Packit 6c0a39
Packit 6c0a39
  if (len == 0) {
Packit 6c0a39
    return strdup(".");
Packit 6c0a39
  } else if (len == 1) {
Packit 6c0a39
    return strdup("/");
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  /* Remove slashes again */
Packit 6c0a39
  while(len > 0 && path[len - 1] == '/') --len;
Packit 6c0a39
Packit 6c0a39
  new = malloc(len + 1);
Packit 6c0a39
  if (new == NULL) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  strncpy(new, path, len);
Packit 6c0a39
  new[len] = '\0';
Packit 6c0a39
Packit 6c0a39
  return new;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief basename - parse filename component.
Packit 6c0a39
 *
Packit 6c0a39
 * basename breaks a null-terminated pathname string into a filename component.
Packit 6c0a39
 * ssh_basename() returns the component following the final '/'.  Trailing '/'
Packit 6c0a39
 * characters are not counted as part of the pathname.
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  path     The path to parse.
Packit 6c0a39
 *
Packit 6c0a39
 * @return              The filename of path or NULL if we can't allocate
Packit 6c0a39
 *                      memory. If path is a the string "/", basename returns
Packit 6c0a39
 *                      the string "/". If path is NULL or an empty string,
Packit 6c0a39
 *                      "." is returned.
Packit 6c0a39
 */
Packit 6c0a39
char *ssh_basename (const char *path) {
Packit 6c0a39
  char *new = NULL;
Packit 6c0a39
  const char *s;
Packit 6c0a39
  size_t len;
Packit 6c0a39
Packit 6c0a39
  if (path == NULL || *path == '\0') {
Packit 6c0a39
    return strdup(".");
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  len = strlen(path);
Packit 6c0a39
  /* Remove trailing slashes */
Packit 6c0a39
  while(len > 0 && path[len - 1] == '/') --len;
Packit 6c0a39
Packit 6c0a39
  /* We have only slashes */
Packit 6c0a39
  if (len == 0) {
Packit 6c0a39
    return strdup("/");
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  while(len > 0 && path[len - 1] != '/') --len;
Packit 6c0a39
Packit 6c0a39
  if (len > 0) {
Packit 6c0a39
    s = path + len;
Packit 6c0a39
    len = strlen(s);
Packit 6c0a39
Packit 6c0a39
    while(len > 0 && s[len - 1] == '/') --len;
Packit 6c0a39
  } else {
Packit 6c0a39
    return strdup(path);
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  new = malloc(len + 1);
Packit 6c0a39
  if (new == NULL) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  strncpy(new, s, len);
Packit 6c0a39
  new[len] = '\0';
Packit 6c0a39
Packit 6c0a39
  return new;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief Attempts to create a directory with the given pathname.
Packit 6c0a39
 *
Packit 6c0a39
 * This is the portable version of mkdir, mode is ignored on Windows systems.
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  pathname The path name to create the directory.
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  mode     The permissions to use.
Packit 6c0a39
 *
Packit 6c0a39
 * @return              0 on success, < 0 on error with errno set.
Packit 6c0a39
 */
Packit Service fcc0d2
int ssh_mkdir(const char *pathname, mode_t mode)
Packit Service fcc0d2
{
Packit Service fcc0d2
    int r;
Packit Service fcc0d2
#ifdef _WIN32
Packit Service fcc0d2
    r = _mkdir(pathname);
Packit Service fcc0d2
#else
Packit Service fcc0d2
    r = mkdir(pathname, mode);
Packit Service fcc0d2
#endif
Packit Service fcc0d2
Packit Service fcc0d2
    return r;
Packit Service fcc0d2
}
Packit Service fcc0d2
Packit Service fcc0d2
/**
Packit Service fcc0d2
 * @brief Attempts to create a directory with the given pathname. The missing
Packit Service fcc0d2
 * directories in the given pathname are created recursively.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in]  pathname The path name to create the directory.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in]  mode     The permissions to use.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @return              0 on success, < 0 on error with errno set.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @note mode is ignored on Windows systems.
Packit Service fcc0d2
 */
Packit Service fcc0d2
int ssh_mkdirs(const char *pathname, mode_t mode)
Packit Service fcc0d2
{
Packit Service fcc0d2
    int rc = 0;
Packit Service fcc0d2
    char *parent = NULL;
Packit Service fcc0d2
Packit Service fcc0d2
    if (pathname == NULL ||
Packit Service fcc0d2
        pathname[0] == '\0' ||
Packit Service fcc0d2
        !strcmp(pathname, "/") ||
Packit Service fcc0d2
        !strcmp(pathname, "."))
Packit Service fcc0d2
    {
Packit Service fcc0d2
        errno = EINVAL;
Packit Service fcc0d2
        return -1;
Packit Service fcc0d2
    }
Packit 6c0a39
Packit Service fcc0d2
    errno = 0;
Packit Service fcc0d2
Packit Service fcc0d2
#ifdef _WIN32
Packit Service fcc0d2
    rc = _mkdir(pathname);
Packit Service fcc0d2
#else
Packit Service fcc0d2
    rc = mkdir(pathname, mode);
Packit Service fcc0d2
#endif
Packit Service fcc0d2
Packit Service fcc0d2
    if (rc < 0) {
Packit Service fcc0d2
        /* If a directory was missing, try to create the parent */
Packit Service fcc0d2
        if (errno == ENOENT) {
Packit Service fcc0d2
            parent = ssh_dirname(pathname);
Packit Service fcc0d2
            if (parent == NULL) {
Packit Service fcc0d2
                errno = ENOMEM;
Packit Service fcc0d2
                return -1;
Packit Service fcc0d2
            }
Packit Service fcc0d2
Packit Service fcc0d2
            rc = ssh_mkdirs(parent, mode);
Packit Service fcc0d2
            if (rc < 0) {
Packit Service fcc0d2
                /* We could not create the parent */
Packit Service fcc0d2
                SAFE_FREE(parent);
Packit Service fcc0d2
                return -1;
Packit Service fcc0d2
            }
Packit Service fcc0d2
Packit Service fcc0d2
            SAFE_FREE(parent);
Packit Service fcc0d2
Packit Service fcc0d2
            /* Try again */
Packit Service fcc0d2
            errno = 0;
Packit 6c0a39
#ifdef _WIN32
Packit Service fcc0d2
            rc = _mkdir(pathname);
Packit 6c0a39
#else
Packit Service fcc0d2
            rc = mkdir(pathname, mode);
Packit 6c0a39
#endif
Packit Service fcc0d2
        }
Packit Service fcc0d2
    }
Packit 6c0a39
Packit Service fcc0d2
    return rc;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief Expand a directory starting with a tilde '~'
Packit 6c0a39
 *
Packit 6c0a39
 * @param[in]  d        The directory to expand.
Packit 6c0a39
 *
Packit 6c0a39
 * @return              The expanded directory, NULL on error.
Packit 6c0a39
 */
Packit 6c0a39
char *ssh_path_expand_tilde(const char *d) {
Packit 6c0a39
    char *h = NULL, *r;
Packit 6c0a39
    const char *p;
Packit 6c0a39
    size_t ld;
Packit 6c0a39
    size_t lh = 0;
Packit 6c0a39
Packit 6c0a39
    if (d[0] != '~') {
Packit 6c0a39
        return strdup(d);
Packit 6c0a39
    }
Packit 6c0a39
    d++;
Packit 6c0a39
Packit 6c0a39
    /* handle ~user/path */
Packit 6c0a39
    p = strchr(d, '/');
Packit 6c0a39
    if (p != NULL && p > d) {
Packit 6c0a39
#ifdef _WIN32
Packit 6c0a39
        return strdup(d);
Packit 6c0a39
#else
Packit 6c0a39
        struct passwd *pw;
Packit 6c0a39
        size_t s = p - d;
Packit 6c0a39
        char u[128];
Packit 6c0a39
Packit 6c0a39
        if (s >= sizeof(u)) {
Packit 6c0a39
            return NULL;
Packit 6c0a39
        }
Packit 6c0a39
        memcpy(u, d, s);
Packit 6c0a39
        u[s] = '\0';
Packit 6c0a39
        pw = getpwnam(u);
Packit 6c0a39
        if (pw == NULL) {
Packit 6c0a39
            return NULL;
Packit 6c0a39
        }
Packit 6c0a39
        ld = strlen(p);
Packit 6c0a39
        h = strdup(pw->pw_dir);
Packit 6c0a39
#endif
Packit 6c0a39
    } else {
Packit 6c0a39
        ld = strlen(d);
Packit 6c0a39
        p = (char *) d;
Packit 6c0a39
        h = ssh_get_user_home_dir();
Packit 6c0a39
    }
Packit 6c0a39
    if (h == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
    lh = strlen(h);
Packit 6c0a39
Packit 6c0a39
    r = malloc(ld + lh + 1);
Packit 6c0a39
    if (r == NULL) {
Packit 6c0a39
        SAFE_FREE(h);
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    if (lh > 0) {
Packit 6c0a39
        memcpy(r, h, lh);
Packit 6c0a39
    }
Packit 6c0a39
    SAFE_FREE(h);
Packit 6c0a39
    memcpy(r + lh, p, ld + 1);
Packit 6c0a39
Packit 6c0a39
    return r;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/** @internal
Packit 6c0a39
 * @brief expands a string in function of session options
Packit 6c0a39
 * @param[in] s Format string to expand. Known parameters:
Packit 6c0a39
 *              %d SSH configuration directory (~/.ssh)
Packit 6c0a39
 *              %h target host name
Packit 6c0a39
 *              %u local username
Packit 6c0a39
 *              %l local hostname
Packit 6c0a39
 *              %r remote username
Packit 6c0a39
 *              %p remote port
Packit 6c0a39
 * @returns Expanded string.
Packit 6c0a39
 */
Packit 6c0a39
char *ssh_path_expand_escape(ssh_session session, const char *s) {
Packit 6c0a39
    char host[NI_MAXHOST];
Packit 6c0a39
    char buf[MAX_BUF_SIZE];
Packit 6c0a39
    char *r, *x = NULL;
Packit 6c0a39
    const char *p;
Packit 6c0a39
    size_t i, l;
Packit 6c0a39
Packit 6c0a39
    r = ssh_path_expand_tilde(s);
Packit 6c0a39
    if (r == NULL) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    if (strlen(r) > MAX_BUF_SIZE) {
Packit 6c0a39
        ssh_set_error(session, SSH_FATAL, "string to expand too long");
Packit 6c0a39
        free(r);
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    p = r;
Packit 6c0a39
    buf[0] = '\0';
Packit 6c0a39
Packit 6c0a39
    for (i = 0; *p != '\0'; p++) {
Packit 6c0a39
        if (*p != '%') {
Packit Service fcc0d2
        escape:
Packit 6c0a39
            buf[i] = *p;
Packit 6c0a39
            i++;
Packit 6c0a39
            if (i >= MAX_BUF_SIZE) {
Packit 6c0a39
                free(r);
Packit 6c0a39
                return NULL;
Packit 6c0a39
            }
Packit 6c0a39
            buf[i] = '\0';
Packit 6c0a39
            continue;
Packit 6c0a39
        }
Packit 6c0a39
Packit 6c0a39
        p++;
Packit 6c0a39
        if (*p == '\0') {
Packit 6c0a39
            break;
Packit 6c0a39
        }
Packit 6c0a39
Packit 6c0a39
        switch (*p) {
Packit Service fcc0d2
            case '%':
Packit Service fcc0d2
                goto escape;
Packit 6c0a39
            case 'd':
Packit 6c0a39
                x = strdup(session->opts.sshdir);
Packit 6c0a39
                break;
Packit 6c0a39
            case 'u':
Packit 6c0a39
                x = ssh_get_local_username();
Packit 6c0a39
                break;
Packit 6c0a39
            case 'l':
Packit 6c0a39
                if (gethostname(host, sizeof(host) == 0)) {
Packit 6c0a39
                    x = strdup(host);
Packit 6c0a39
                }
Packit 6c0a39
                break;
Packit 6c0a39
            case 'h':
Packit 6c0a39
                x = strdup(session->opts.host);
Packit 6c0a39
                break;
Packit 6c0a39
            case 'r':
Packit 6c0a39
                x = strdup(session->opts.username);
Packit 6c0a39
                break;
Packit 6c0a39
            case 'p':
Packit 6c0a39
                if (session->opts.port < 65536) {
Packit 6c0a39
                    char tmp[6];
Packit 6c0a39
Packit 6c0a39
                    snprintf(tmp,
Packit 6c0a39
                             sizeof(tmp),
Packit 6c0a39
                             "%u",
Packit 6c0a39
                             session->opts.port > 0 ? session->opts.port : 22);
Packit 6c0a39
                    x = strdup(tmp);
Packit 6c0a39
                }
Packit 6c0a39
                break;
Packit 6c0a39
            default:
Packit 6c0a39
                ssh_set_error(session, SSH_FATAL,
Packit 6c0a39
                        "Wrong escape sequence detected");
Packit 6c0a39
                free(r);
Packit 6c0a39
                return NULL;
Packit 6c0a39
        }
Packit 6c0a39
Packit 6c0a39
        if (x == NULL) {
Packit 6c0a39
            ssh_set_error_oom(session);
Packit 6c0a39
            free(r);
Packit 6c0a39
            return NULL;
Packit 6c0a39
        }
Packit 6c0a39
Packit 6c0a39
        i += strlen(x);
Packit 6c0a39
        if (i >= MAX_BUF_SIZE) {
Packit 6c0a39
            ssh_set_error(session, SSH_FATAL,
Packit 6c0a39
                    "String too long");
Packit 6c0a39
            free(x);
Packit 6c0a39
            free(r);
Packit 6c0a39
            return NULL;
Packit 6c0a39
        }
Packit 6c0a39
        l = strlen(buf);
Packit 6c0a39
        strncpy(buf + l, x, sizeof(buf) - l - 1);
Packit 6c0a39
        buf[i] = '\0';
Packit 6c0a39
        SAFE_FREE(x);
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    free(r);
Packit 6c0a39
    return strdup(buf);
Packit 6c0a39
#undef MAX_BUF_SIZE
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @internal
Packit 6c0a39
 *
Packit 6c0a39
 * @brief Analyze the SSH banner to extract version information.
Packit 6c0a39
 *
Packit 6c0a39
 * @param  session      The session to analyze the banner from.
Packit 6c0a39
 * @param  server       0 means we are a client, 1 a server.
Packit 6c0a39
 *
Packit 6c0a39
 * @return 0 on success, < 0 on error.
Packit 6c0a39
 *
Packit 6c0a39
 * @see ssh_get_issue_banner()
Packit 6c0a39
 */
Packit 6c0a39
int ssh_analyze_banner(ssh_session session, int server)
Packit 6c0a39
{
Packit 6c0a39
    const char *banner;
Packit 6c0a39
    const char *openssh;
Packit 6c0a39
Packit 6c0a39
    if (server) {
Packit 6c0a39
        banner = session->clientbanner;
Packit 6c0a39
    } else {
Packit 6c0a39
        banner = session->serverbanner;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    if (banner == NULL) {
Packit 6c0a39
        ssh_set_error(session, SSH_FATAL, "Invalid banner");
Packit 6c0a39
        return -1;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    /*
Packit 6c0a39
     * Typical banners e.g. are:
Packit 6c0a39
     *
Packit 6c0a39
     * SSH-1.5-openSSH_5.4
Packit 6c0a39
     * SSH-1.99-openSSH_3.0
Packit 6c0a39
     *
Packit 6c0a39
     * SSH-2.0-something
Packit 6c0a39
     * 012345678901234567890
Packit 6c0a39
     */
Packit 6c0a39
    if (strlen(banner) < 6 ||
Packit 6c0a39
        strncmp(banner, "SSH-", 4) != 0) {
Packit 6c0a39
          ssh_set_error(session, SSH_FATAL, "Protocol mismatch: %s", banner);
Packit 6c0a39
          return -1;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    SSH_LOG(SSH_LOG_PROTOCOL, "Analyzing banner: %s", banner);
Packit 6c0a39
Packit 6c0a39
    switch (banner[4]) {
Packit 6c0a39
        case '2':
Packit 6c0a39
            break;
Packit 6c0a39
        case '1':
Packit 6c0a39
            if (strlen(banner) > 6) {
Packit 6c0a39
                if (banner[6] == '9') {
Packit 6c0a39
                    break;
Packit 6c0a39
                }
Packit 6c0a39
            }
Packit 6c0a39
            FALL_THROUGH;
Packit 6c0a39
        default:
Packit 6c0a39
            ssh_set_error(session, SSH_FATAL, "Protocol mismatch: %s", banner);
Packit 6c0a39
            return -1;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    /* Make a best-effort to extract OpenSSH version numbers. */
Packit 6c0a39
    openssh = strstr(banner, "OpenSSH");
Packit 6c0a39
    if (openssh != NULL) {
Packit 6c0a39
        char *tmp = NULL;
Packit 6c0a39
        unsigned long int major = 0UL;
Packit 6c0a39
        unsigned long int minor = 0UL;
Packit 6c0a39
Packit 6c0a39
        /*
Packit 6c0a39
         * The banner is typical:
Packit 6c0a39
         * OpenSSH_5.4
Packit 6c0a39
         * 012345678901234567890
Packit 6c0a39
         */
Packit 6c0a39
        if (strlen(openssh) > 9) {
Packit 6c0a39
            major = strtoul(openssh + 8, &tmp, 10);
Packit 6c0a39
            if ((tmp == (openssh + 8)) ||
Packit 6c0a39
                ((errno == ERANGE) && (major == ULONG_MAX)) ||
Packit 6c0a39
                ((errno != 0) && (major == 0)) ||
Packit 6c0a39
                ((major < 1) || (major > 100))) {
Packit 6c0a39
                /* invalid major */
Packit 6c0a39
                goto done;
Packit 6c0a39
            }
Packit 6c0a39
Packit 6c0a39
            minor = strtoul(openssh + 10, &tmp, 10);
Packit 6c0a39
            if ((tmp == (openssh + 10)) ||
Packit 6c0a39
                ((errno == ERANGE) && (major == ULONG_MAX)) ||
Packit 6c0a39
                ((errno != 0) && (major == 0)) ||
Packit 6c0a39
                (minor > 100)) {
Packit 6c0a39
                /* invalid minor */
Packit 6c0a39
                goto done;
Packit 6c0a39
            }
Packit 6c0a39
Packit 6c0a39
            session->openssh = SSH_VERSION_INT(((int) major), ((int) minor), 0);
Packit 6c0a39
Packit 6c0a39
            SSH_LOG(SSH_LOG_PROTOCOL,
Packit 6c0a39
                    "We are talking to an OpenSSH client version: %lu.%lu (%x)",
Packit 6c0a39
                    major, minor, session->openssh);
Packit 6c0a39
        }
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
done:
Packit 6c0a39
    return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/* try the Monotonic clock if possible for perfs reasons */
Packit 6c0a39
#ifdef _POSIX_MONOTONIC_CLOCK
Packit 6c0a39
#define CLOCK CLOCK_MONOTONIC
Packit 6c0a39
#else
Packit 6c0a39
#define CLOCK CLOCK_REALTIME
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @internal
Packit 6c0a39
 * @brief initializes a timestamp to the current time
Packit 6c0a39
 * @param[out] ts pointer to an allocated ssh_timestamp structure
Packit 6c0a39
 */
Packit 6c0a39
void ssh_timestamp_init(struct ssh_timestamp *ts){
Packit 6c0a39
#ifdef HAVE_CLOCK_GETTIME
Packit 6c0a39
  struct timespec tp;
Packit 6c0a39
  clock_gettime(CLOCK, &tp);
Packit 6c0a39
  ts->useconds = tp.tv_nsec / 1000;
Packit 6c0a39
#else
Packit 6c0a39
  struct timeval tp;
Packit 6c0a39
  gettimeofday(&tp, NULL);
Packit 6c0a39
  ts->useconds = tp.tv_usec;
Packit 6c0a39
#endif
Packit 6c0a39
  ts->seconds = tp.tv_sec;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
#undef CLOCK
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @internal
Packit 6c0a39
 * @brief gets the time difference between two timestamps in ms
Packit 6c0a39
 * @param[in] old older value
Packit 6c0a39
 * @param[in] new newer value
Packit 6c0a39
 * @returns difference in milliseconds
Packit 6c0a39
 */
Packit 6c0a39
Packit 6c0a39
static int ssh_timestamp_difference(struct ssh_timestamp *old,
Packit 6c0a39
    struct ssh_timestamp *new){
Packit 6c0a39
  long seconds, usecs, msecs;
Packit 6c0a39
  seconds = new->seconds - old->seconds;
Packit 6c0a39
  usecs = new->useconds - old->useconds;
Packit 6c0a39
  if (usecs < 0){
Packit 6c0a39
    seconds--;
Packit 6c0a39
    usecs += 1000000;
Packit 6c0a39
  }
Packit 6c0a39
  msecs = seconds * 1000 + usecs/1000;
Packit 6c0a39
  return msecs;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @internal
Packit 6c0a39
 * @brief turn seconds and microseconds pair (as provided by user-set options)
Packit 6c0a39
 * into millisecond value
Packit 6c0a39
 * @param[in] sec number of seconds
Packit 6c0a39
 * @param[in] usec number of microseconds
Packit 6c0a39
 * @returns milliseconds, or 10000 if user supplied values are equal to zero
Packit 6c0a39
 */
Packit 6c0a39
int ssh_make_milliseconds(long sec, long usec) {
Packit 6c0a39
	int res = usec ? (usec / 1000) : 0;
Packit 6c0a39
	res += (sec * 1000);
Packit 6c0a39
	if (res == 0) {
Packit 6c0a39
		res = 10 * 1000; /* use a reasonable default value in case
Packit 6c0a39
				* SSH_OPTIONS_TIMEOUT is not set in options. */
Packit 6c0a39
	}
Packit 6c0a39
	return res;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @internal
Packit 6c0a39
 * @brief Checks if a timeout is elapsed, in function of a previous
Packit 6c0a39
 * timestamp and an assigned timeout
Packit 6c0a39
 * @param[in] ts pointer to an existing timestamp
Packit 6c0a39
 * @param[in] timeout timeout in milliseconds. Negative values mean infinite
Packit 6c0a39
 *                   timeout
Packit 6c0a39
 * @returns 1 if timeout is elapsed
Packit 6c0a39
 *          0 otherwise
Packit 6c0a39
 */
Packit 6c0a39
int ssh_timeout_elapsed(struct ssh_timestamp *ts, int timeout) {
Packit 6c0a39
    struct ssh_timestamp now;
Packit 6c0a39
Packit 6c0a39
    switch(timeout) {
Packit 6c0a39
        case -2: /*
Packit 6c0a39
                  * -2 means user-defined timeout as available in
Packit 6c0a39
                  * session->timeout, session->timeout_usec.
Packit 6c0a39
                  */
Packit 6c0a39
            SSH_LOG(SSH_LOG_WARN, "ssh_timeout_elapsed called with -2. this needs to "
Packit 6c0a39
                            "be fixed. please set a breakpoint on misc.c:%d and "
Packit 6c0a39
                            "fix the caller\n", __LINE__);
Packit 6c0a39
            return 0;
Packit 6c0a39
        case -1: /* -1 means infinite timeout */
Packit 6c0a39
            return 0;
Packit 6c0a39
        case 0: /* 0 means no timeout */
Packit 6c0a39
            return 1;
Packit 6c0a39
        default:
Packit 6c0a39
            break;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    ssh_timestamp_init(&now;;
Packit 6c0a39
Packit 6c0a39
    return (ssh_timestamp_difference(ts,&now) >= timeout);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/**
Packit 6c0a39
 * @brief updates a timeout value so it reflects the remaining time
Packit 6c0a39
 * @param[in] ts pointer to an existing timestamp
Packit 6c0a39
 * @param[in] timeout timeout in milliseconds. Negative values mean infinite
Packit 6c0a39
 *             timeout
Packit 6c0a39
 * @returns   remaining time in milliseconds, 0 if elapsed, -1 if never.
Packit 6c0a39
 */
Packit 6c0a39
int ssh_timeout_update(struct ssh_timestamp *ts, int timeout){
Packit 6c0a39
  struct ssh_timestamp now;
Packit 6c0a39
  int ms, ret;
Packit 6c0a39
  if (timeout <= 0) {
Packit 6c0a39
      return timeout;
Packit 6c0a39
  }
Packit 6c0a39
  ssh_timestamp_init(&now;;
Packit 6c0a39
  ms = ssh_timestamp_difference(ts,&now;;
Packit 6c0a39
  if(ms < 0)
Packit 6c0a39
    ms = 0;
Packit 6c0a39
  ret = timeout - ms;
Packit 6c0a39
  return ret >= 0 ? ret: 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
Packit 6c0a39
int ssh_match_group(const char *group, const char *object)
Packit 6c0a39
{
Packit 6c0a39
    const char *a;
Packit 6c0a39
    const char *z;
Packit 6c0a39
Packit 6c0a39
    z = group;
Packit 6c0a39
    do {
Packit 6c0a39
        a = strchr(z, ',');
Packit 6c0a39
        if (a == NULL) {
Packit 6c0a39
            if (strcmp(z, object) == 0) {
Packit 6c0a39
                return 1;
Packit 6c0a39
            }
Packit 6c0a39
            return 0;
Packit 6c0a39
        } else {
Packit 6c0a39
            if (strncmp(z, object, a - z) == 0) {
Packit 6c0a39
                return 1;
Packit 6c0a39
            }
Packit 6c0a39
        }
Packit 6c0a39
        z = a + 1;
Packit 6c0a39
    } while(1);
Packit 6c0a39
Packit 6c0a39
    /* not reached */
Packit 6c0a39
    return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
#if !defined(HAVE_EXPLICIT_BZERO)
Packit 6c0a39
void explicit_bzero(void *s, size_t n)
Packit 6c0a39
{
Packit 6c0a39
#if defined(HAVE_MEMSET_S)
Packit 6c0a39
    memset_s(s, n, '\0', n);
Packit 6c0a39
#elif defined(HAVE_SECURE_ZERO_MEMORY)
Packit 6c0a39
    SecureZeroMemory(s, n);
Packit 6c0a39
#else
Packit 6c0a39
    memset(s, '\0', n);
Packit 6c0a39
#if defined(HAVE_GCC_VOLATILE_MEMORY_PROTECTION)
Packit 6c0a39
    /* See http://llvm.org/bugs/show_bug.cgi?id=15495 */
Packit 6c0a39
    __asm__ volatile("" : : "g"(s) : "memory");
Packit 6c0a39
#endif /* HAVE_GCC_VOLATILE_MEMORY_PROTECTION */
Packit 6c0a39
#endif
Packit 6c0a39
}
Packit 6c0a39
#endif /* !HAVE_EXPLICIT_BZERO */
Packit 6c0a39
Packit 6c0a39
#if !defined(HAVE_STRNDUP)
Packit 6c0a39
char *strndup(const char *s, size_t n)
Packit 6c0a39
{
Packit 6c0a39
    char *x = NULL;
Packit 6c0a39
Packit 6c0a39
    if (n + 1 < n) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    x = malloc(n + 1);
Packit 6c0a39
    if (x == NULL) {
Packit 6c0a39
        return NULL;
Packit 6c0a39
    }
Packit 6c0a39
Packit 6c0a39
    memcpy(x, s, n);
Packit 6c0a39
    x[n] = '\0';
Packit 6c0a39
Packit 6c0a39
    return x;
Packit 6c0a39
}
Packit 6c0a39
#endif /* ! HAVE_STRNDUP */
Packit 6c0a39
Packit 6c0a39
/* Increment 64b integer in network byte order */
Packit 6c0a39
void
Packit 6c0a39
uint64_inc(unsigned char *counter)
Packit 6c0a39
{
Packit 6c0a39
    int i;
Packit 6c0a39
Packit 6c0a39
    for (i = 7; i >= 0; i--) {
Packit 6c0a39
        counter[i]++;
Packit 6c0a39
        if (counter[i])
Packit 6c0a39
          return;
Packit 6c0a39
    }
Packit 6c0a39
}
Packit 6c0a39
Packit Service fcc0d2
/**
Packit Service fcc0d2
 * @internal
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @brief Quote file name to be used on shell.
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * Try to put the given file name between single quotes. There are special
Packit Service fcc0d2
 * cases:
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * - When the '\'' char is found in the file name, it is double quoted
Packit Service fcc0d2
 *   - example:
Packit Service fcc0d2
 *     input: a'b
Packit Service fcc0d2
 *     output: 'a'"'"'b'
Packit Service fcc0d2
 * - When the '!' char is found in the file name, it is replaced by an unquoted
Packit Service fcc0d2
 *   verbatim char "\!"
Packit Service fcc0d2
 *   - example:
Packit Service fcc0d2
 *     input: a!b
Packit Service fcc0d2
 *     output 'a'\!'b'
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in]   file_name  File name string to be quoted before used on shell
Packit Service fcc0d2
 * @param[out]  buf       Buffer to receive the final quoted file name.  Must
Packit Service fcc0d2
 *                        have room for the final quoted string.  The maximum
Packit Service fcc0d2
 *                        output length would be (3 * strlen(file_name) + 1)
Packit Service fcc0d2
 *                        since in the worst case each character would be
Packit Service fcc0d2
 *                        replaced by 3 characters, plus the terminating '\0'.
Packit Service fcc0d2
 * @param[in]   buf_len   The size of the provided output buffer
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @returns SSH_ERROR on error; length of the resulting string not counting the
Packit Service fcc0d2
 * string terminator '\0'
Packit Service fcc0d2
 * */
Packit Service fcc0d2
int ssh_quote_file_name(const char *file_name, char *buf, size_t buf_len)
Packit Service fcc0d2
{
Packit Service fcc0d2
    const char *src = NULL;
Packit Service fcc0d2
    char *dst = NULL;
Packit Service fcc0d2
    size_t required_buf_len;
Packit Service fcc0d2
Packit Service fcc0d2
    enum ssh_quote_state_e state = NO_QUOTE;
Packit Service fcc0d2
Packit Service fcc0d2
    if (file_name == NULL || buf == NULL || buf_len == 0) {
Packit Service fcc0d2
        SSH_LOG(SSH_LOG_WARNING, "Invalid parameter");
Packit Service fcc0d2
        return SSH_ERROR;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    /* Only allow file names smaller than 32kb. */
Packit Service fcc0d2
    if (strlen(file_name) > 32 * 1024) {
Packit Service fcc0d2
        SSH_LOG(SSH_LOG_WARNING, "File name too long");
Packit Service fcc0d2
        return SSH_ERROR;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    /* Paranoia check */
Packit Service fcc0d2
    required_buf_len = (size_t)3 * strlen(file_name) + 1;
Packit Service fcc0d2
    if (required_buf_len > buf_len) {
Packit Service fcc0d2
        SSH_LOG(SSH_LOG_WARNING, "Buffer too small");
Packit Service fcc0d2
        return SSH_ERROR;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    src = file_name;
Packit Service fcc0d2
    dst = buf;
Packit Service fcc0d2
Packit Service fcc0d2
    while ((*src != '\0')) {
Packit Service fcc0d2
        switch (*src) {
Packit Service fcc0d2
Packit Service fcc0d2
        /* The '\'' char is double quoted */
Packit Service fcc0d2
Packit Service fcc0d2
        case '\'':
Packit Service fcc0d2
            switch (state) {
Packit Service fcc0d2
            case NO_QUOTE:
Packit Service fcc0d2
                /* Start a new double quoted string. The '\'' char will be
Packit Service fcc0d2
                 * copied to the beginning of it at the end of the loop. */
Packit Service fcc0d2
                *dst++ = '"';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            case SINGLE_QUOTE:
Packit Service fcc0d2
                /* Close the current single quoted string and start a new double
Packit Service fcc0d2
                 * quoted string. The '\'' char will be copied to the beginning
Packit Service fcc0d2
                 * of it at the end of the loop. */
Packit Service fcc0d2
                *dst++ = '\'';
Packit Service fcc0d2
                *dst++ = '"';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            case DOUBLE_QUOTE:
Packit Service fcc0d2
                /* If already in the double quoted string, keep copying the
Packit Service fcc0d2
                 * sequence of chars. */
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            default:
Packit Service fcc0d2
                /* Should never be reached */
Packit Service fcc0d2
                goto error;
Packit Service fcc0d2
            }
Packit Service fcc0d2
Packit Service fcc0d2
            /* When the '\'' char is found, the resulting state will be
Packit Service fcc0d2
             * DOUBLE_QUOTE in any case*/
Packit Service fcc0d2
            state = DOUBLE_QUOTE;
Packit Service fcc0d2
            break;
Packit Service fcc0d2
Packit Service fcc0d2
        /* The '!' char is replaced by unquoted "\!" */
Packit Service fcc0d2
Packit Service fcc0d2
        case '!':
Packit Service fcc0d2
            switch (state) {
Packit Service fcc0d2
            case NO_QUOTE:
Packit Service fcc0d2
                /* The '!' char is interpreted in some shells (e.g. CSH) even
Packit Service fcc0d2
                 * when is quoted with single quotes.  Replace it with unquoted
Packit Service fcc0d2
                 * "\!" which is correctly interpreted as the '!' character. */
Packit Service fcc0d2
                *dst++ = '\\';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            case SINGLE_QUOTE:
Packit Service fcc0d2
                /* Close the current quoted string and replace '!' for unquoted
Packit Service fcc0d2
                 * "\!" */
Packit Service fcc0d2
                *dst++ = '\'';
Packit Service fcc0d2
                *dst++ = '\\';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            case DOUBLE_QUOTE:
Packit Service fcc0d2
                /* Close current quoted string and replace  "!" for unquoted
Packit Service fcc0d2
                 * "\!" */
Packit Service fcc0d2
                *dst++ = '"';
Packit Service fcc0d2
                *dst++ = '\\';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            default:
Packit Service fcc0d2
                /* Should never be reached */
Packit Service fcc0d2
                goto error;
Packit Service fcc0d2
            }
Packit Service fcc0d2
Packit Service fcc0d2
            /* When the '!' char is found, the resulting state will be NO_QUOTE
Packit Service fcc0d2
             * in any case*/
Packit Service fcc0d2
            state = NO_QUOTE;
Packit Service fcc0d2
            break;
Packit Service fcc0d2
Packit Service fcc0d2
        /* Ordinary chars are single quoted */
Packit Service fcc0d2
Packit Service fcc0d2
        default:
Packit Service fcc0d2
            switch (state) {
Packit Service fcc0d2
            case NO_QUOTE:
Packit Service fcc0d2
                /* Start a new single quoted string */
Packit Service fcc0d2
                *dst++ = '\'';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            case SINGLE_QUOTE:
Packit Service fcc0d2
                /* If already in the single quoted string, keep copying the
Packit Service fcc0d2
                 * sequence of chars. */
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            case DOUBLE_QUOTE:
Packit Service fcc0d2
                /* Close current double quoted string and start a new single
Packit Service fcc0d2
                 * quoted string. */
Packit Service fcc0d2
                *dst++ = '"';
Packit Service fcc0d2
                *dst++ = '\'';
Packit Service fcc0d2
                break;
Packit Service fcc0d2
            default:
Packit Service fcc0d2
                /* Should never be reached */
Packit Service fcc0d2
                goto error;
Packit Service fcc0d2
            }
Packit Service fcc0d2
Packit Service fcc0d2
            /* When an ordinary char is found, the resulting state will be
Packit Service fcc0d2
             * SINGLE_QUOTE in any case*/
Packit Service fcc0d2
            state = SINGLE_QUOTE;
Packit Service fcc0d2
            break;
Packit Service fcc0d2
        }
Packit Service fcc0d2
Packit Service fcc0d2
        /* Copy the current char to output */
Packit Service fcc0d2
        *dst++ = *src++;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    /* Close the quoted string when necessary */
Packit Service fcc0d2
Packit Service fcc0d2
    switch (state) {
Packit Service fcc0d2
    case NO_QUOTE:
Packit Service fcc0d2
        /* No open string */
Packit Service fcc0d2
        break;
Packit Service fcc0d2
    case SINGLE_QUOTE:
Packit Service fcc0d2
        /* Close current single quoted string */
Packit Service fcc0d2
        *dst++ = '\'';
Packit Service fcc0d2
        break;
Packit Service fcc0d2
    case DOUBLE_QUOTE:
Packit Service fcc0d2
        /* Close current double quoted string */
Packit Service fcc0d2
        *dst++ = '"';
Packit Service fcc0d2
        break;
Packit Service fcc0d2
    default:
Packit Service fcc0d2
        /* Should never be reached */
Packit Service fcc0d2
        goto error;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    /* Put the string terminator */
Packit Service fcc0d2
    *dst = '\0';
Packit Service fcc0d2
Packit Service fcc0d2
    return dst - buf;
Packit Service fcc0d2
Packit Service fcc0d2
error:
Packit Service fcc0d2
    return SSH_ERROR;
Packit Service fcc0d2
}
Packit Service fcc0d2
Packit Service fcc0d2
/**
Packit Service fcc0d2
 * @internal
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @brief Given a string, encode existing newlines as the string "\\n"
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @param[in]  string   Input string
Packit Service fcc0d2
 * @param[out] buf      Output buffer. This buffer must be at least (2 *
Packit Service fcc0d2
 *                      strlen(string)) + 1 long.  In the worst case,
Packit Service fcc0d2
 *                      each character can be encoded as 2 characters plus the
Packit Service fcc0d2
 *                      terminating '\0'.
Packit Service fcc0d2
 * @param[in]  buf_len  Size of the provided output buffer
Packit Service fcc0d2
 *
Packit Service fcc0d2
 * @returns SSH_ERROR on error; length of the resulting string not counting the
Packit Service fcc0d2
 * terminating '\0' otherwise
Packit Service fcc0d2
 */
Packit Service fcc0d2
int ssh_newline_vis(const char *string, char *buf, size_t buf_len)
Packit Service fcc0d2
{
Packit Service fcc0d2
    const char *in = NULL;
Packit Service fcc0d2
    char *out = NULL;
Packit Service fcc0d2
Packit Service fcc0d2
    if (string == NULL || buf == NULL || buf_len == 0) {
Packit Service fcc0d2
        return SSH_ERROR;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    if ((2 * strlen(string) + 1) > buf_len) {
Packit Service fcc0d2
        SSH_LOG(SSH_LOG_WARNING, "Buffer too small");
Packit Service fcc0d2
        return SSH_ERROR;
Packit Service fcc0d2
    }
Packit Service fcc0d2
Packit Service fcc0d2
    out = buf;
Packit Service fcc0d2
    for (in = string; *in != '\0'; in++) {
Packit Service fcc0d2
        if (*in == '\n') {
Packit Service fcc0d2
            *out++ = '\\';
Packit Service fcc0d2
            *out++ = 'n';
Packit Service fcc0d2
        } else {
Packit Service fcc0d2
            *out++ = *in;
Packit Service fcc0d2
        }
Packit Service fcc0d2
    }
Packit Service fcc0d2
    *out = '\0';
Packit Service fcc0d2
Packit Service fcc0d2
    return out - buf;
Packit Service fcc0d2
}
Packit Service fcc0d2
Packit 6c0a39
/** @} */