Blame src/sftpserver.c

Packit 6c0a39
/*
Packit 6c0a39
 * sftpserver.c - server based function for the sftp protocol
Packit 6c0a39
 *
Packit 6c0a39
 * This file is part of the SSH Library
Packit 6c0a39
 *
Packit 6c0a39
 * Copyright (c) 2005      by Aris Adamantiadis
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
#include <stdio.h>
Packit 6c0a39
Packit 6c0a39
#ifndef _WIN32
Packit 6c0a39
#include <netinet/in.h>
Packit 6c0a39
#include <arpa/inet.h>
Packit 6c0a39
#endif
Packit 6c0a39
Packit 6c0a39
#include "libssh/libssh.h"
Packit 6c0a39
#include "libssh/sftp.h"
Packit 6c0a39
#include "libssh/ssh2.h"
Packit 6c0a39
#include "libssh/priv.h"
Packit 6c0a39
#include "libssh/buffer.h"
Packit 6c0a39
#include "libssh/misc.h"
Packit 6c0a39
Packit 6c0a39
sftp_client_message sftp_get_client_message(sftp_session sftp) {
Packit 6c0a39
  ssh_session session = sftp->session;
Packit 6c0a39
  sftp_packet packet;
Packit 6c0a39
  sftp_client_message msg;
Packit 6c0a39
  ssh_buffer payload;
Packit 6c0a39
  int rc;
Packit 6c0a39
Packit 6c0a39
  msg = malloc(sizeof (struct sftp_client_message_struct));
Packit 6c0a39
  if (msg == NULL) {
Packit 6c0a39
    ssh_set_error_oom(session);
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
  ZERO_STRUCTP(msg);
Packit 6c0a39
Packit 6c0a39
  packet = sftp_packet_read(sftp);
Packit 6c0a39
  if (packet == NULL) {
Packit 6c0a39
    ssh_set_error_oom(session);
Packit 6c0a39
    sftp_client_message_free(msg);
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  payload = packet->payload;
Packit 6c0a39
  msg->type = packet->type;
Packit 6c0a39
  msg->sftp = sftp;
Packit 6c0a39
Packit 6c0a39
  /* take a copy of the whole packet */
Packit 6c0a39
  msg->complete_message = ssh_buffer_new();
Packit 6c0a39
  ssh_buffer_add_data(msg->complete_message,
Packit 6c0a39
                      ssh_buffer_get(payload),
Packit 6c0a39
                      ssh_buffer_get_len(payload));
Packit 6c0a39
Packit 6c0a39
  ssh_buffer_get_u32(payload, &msg->id);
Packit 6c0a39
Packit 6c0a39
  switch(msg->type) {
Packit 6c0a39
    case SSH_FXP_CLOSE:
Packit 6c0a39
    case SSH_FXP_READDIR:
Packit 6c0a39
      msg->handle = ssh_buffer_get_ssh_string(payload);
Packit 6c0a39
      if (msg->handle == NULL) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_READ:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "Sqd",
Packit 6c0a39
                             &msg->handle,
Packit 6c0a39
                             &msg->offset,
Packit 6c0a39
                             &msg->len);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_WRITE:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "SqS",
Packit 6c0a39
                             &msg->handle,
Packit 6c0a39
                             &msg->offset,
Packit 6c0a39
                             &msg->data);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_REMOVE:
Packit 6c0a39
    case SSH_FXP_RMDIR:
Packit 6c0a39
    case SSH_FXP_OPENDIR:
Packit 6c0a39
    case SSH_FXP_READLINK:
Packit 6c0a39
    case SSH_FXP_REALPATH:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "s",
Packit 6c0a39
                             &msg->filename);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_RENAME:
Packit 6c0a39
    case SSH_FXP_SYMLINK:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "sS",
Packit 6c0a39
                             &msg->filename,
Packit 6c0a39
                             &msg->data);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_MKDIR:
Packit 6c0a39
    case SSH_FXP_SETSTAT:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "s",
Packit 6c0a39
                             &msg->filename);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      msg->attr = sftp_parse_attr(sftp, payload, 0);
Packit 6c0a39
      if (msg->attr == NULL) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_FSETSTAT:
Packit 6c0a39
      msg->handle = ssh_buffer_get_ssh_string(payload);
Packit 6c0a39
      if (msg->handle == NULL) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      msg->attr = sftp_parse_attr(sftp, payload, 0);
Packit 6c0a39
      if (msg->attr == NULL) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_LSTAT:
Packit 6c0a39
    case SSH_FXP_STAT:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "s",
Packit 6c0a39
                             &msg->filename);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      if(sftp->version > 3) {
Packit 6c0a39
        ssh_buffer_unpack(payload, "d", &msg->flags);
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_OPEN:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "sd",
Packit 6c0a39
                             &msg->filename,
Packit 6c0a39
                             &msg->flags);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      msg->attr = sftp_parse_attr(sftp, payload, 0);
Packit 6c0a39
      if (msg->attr == NULL) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_FSTAT:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "S",
Packit 6c0a39
                             &msg->handle);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    case SSH_FXP_EXTENDED:
Packit 6c0a39
      rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                             "s",
Packit 6c0a39
                             &msg->submessage);
Packit 6c0a39
      if (rc != SSH_OK) {
Packit 6c0a39
        ssh_set_error_oom(session);
Packit 6c0a39
        sftp_client_message_free(msg);
Packit 6c0a39
        return NULL;
Packit 6c0a39
      }
Packit 6c0a39
Packit 6c0a39
      if (strcmp(msg->submessage, "hardlink@openssh.com") == 0 ||
Packit 6c0a39
          strcmp(msg->submessage, "posix-rename@openssh.com") == 0) {
Packit 6c0a39
        rc = ssh_buffer_unpack(payload,
Packit 6c0a39
                               "sS",
Packit 6c0a39
                               &msg->filename,
Packit 6c0a39
                               &msg->data);
Packit 6c0a39
        if (rc != SSH_OK) {
Packit 6c0a39
          ssh_set_error_oom(session);
Packit 6c0a39
          sftp_client_message_free(msg);
Packit 6c0a39
          return NULL;
Packit 6c0a39
        }
Packit 6c0a39
      }
Packit 6c0a39
      break;
Packit 6c0a39
    default:
Packit 6c0a39
      ssh_set_error(sftp->session, SSH_FATAL,
Packit 6c0a39
                    "Received unhandled sftp message %d", msg->type);
Packit 6c0a39
      sftp_client_message_free(msg);
Packit 6c0a39
      return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return msg;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/* Send an sftp client message. Can be used in cas of proxying */
Packit 6c0a39
int sftp_send_client_message(sftp_session sftp, sftp_client_message msg){
Packit 6c0a39
	return sftp_packet_write(sftp, msg->type, msg->complete_message);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
uint8_t sftp_client_message_get_type(sftp_client_message msg){
Packit 6c0a39
	return msg->type;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
const char *sftp_client_message_get_filename(sftp_client_message msg){
Packit 6c0a39
	return msg->filename;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
void sftp_client_message_set_filename(sftp_client_message msg, const char *newname){
Packit 6c0a39
	free(msg->filename);
Packit 6c0a39
	msg->filename = strdup(newname);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
const char *sftp_client_message_get_data(sftp_client_message msg){
Packit 6c0a39
	if (msg->str_data == NULL)
Packit 6c0a39
		msg->str_data = ssh_string_to_char(msg->data);
Packit 6c0a39
	return msg->str_data;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
uint32_t sftp_client_message_get_flags(sftp_client_message msg){
Packit 6c0a39
	return msg->flags;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
const char *sftp_client_message_get_submessage(sftp_client_message msg){
Packit 6c0a39
        return msg->submessage;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
void sftp_client_message_free(sftp_client_message msg) {
Packit 6c0a39
  if (msg == NULL) {
Packit 6c0a39
    return;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  SAFE_FREE(msg->filename);
Packit 6c0a39
  SAFE_FREE(msg->submessage);
Packit 6c0a39
  ssh_string_free(msg->data);
Packit 6c0a39
  ssh_string_free(msg->handle);
Packit 6c0a39
  sftp_attributes_free(msg->attr);
Packit 6c0a39
  ssh_buffer_free(msg->complete_message);
Packit 6c0a39
  SAFE_FREE(msg->str_data);
Packit 6c0a39
  ZERO_STRUCTP(msg);
Packit 6c0a39
  SAFE_FREE(msg);
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_name(sftp_client_message msg, const char *name,
Packit 6c0a39
    sftp_attributes attr) {
Packit 6c0a39
  ssh_buffer out;
Packit 6c0a39
  ssh_string file;
Packit 6c0a39
Packit 6c0a39
  out = ssh_buffer_new();
Packit 6c0a39
  if (out == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  file = ssh_string_from_char(name);
Packit 6c0a39
  if (file == NULL) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_u32(out, msg->id) < 0 ||
Packit 6c0a39
      ssh_buffer_add_u32(out, htonl(1)) < 0 ||
Packit 6c0a39
      ssh_buffer_add_ssh_string(out, file) < 0 ||
Packit 6c0a39
      ssh_buffer_add_ssh_string(out, file) < 0 || /* The protocol is broken here between 3 & 4 */
Packit 6c0a39
      buffer_add_attributes(out, attr) < 0 ||
Packit 6c0a39
      sftp_packet_write(msg->sftp, SSH_FXP_NAME, out) < 0) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    ssh_string_free(file);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
  ssh_buffer_free(out);
Packit 6c0a39
  ssh_string_free(file);
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_handle(sftp_client_message msg, ssh_string handle){
Packit 6c0a39
  ssh_buffer out;
Packit 6c0a39
Packit 6c0a39
  out = ssh_buffer_new();
Packit 6c0a39
  if (out == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_u32(out, msg->id) < 0 ||
Packit 6c0a39
      ssh_buffer_add_ssh_string(out, handle) < 0 ||
Packit 6c0a39
      sftp_packet_write(msg->sftp, SSH_FXP_HANDLE, out) < 0) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
  ssh_buffer_free(out);
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_attr(sftp_client_message msg, sftp_attributes attr) {
Packit 6c0a39
  ssh_buffer out;
Packit 6c0a39
Packit 6c0a39
  out = ssh_buffer_new();
Packit 6c0a39
  if (out == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_u32(out, msg->id) < 0 ||
Packit 6c0a39
      buffer_add_attributes(out, attr) < 0 ||
Packit 6c0a39
      sftp_packet_write(msg->sftp, SSH_FXP_ATTRS, out) < 0) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
  ssh_buffer_free(out);
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_names_add(sftp_client_message msg, const char *file,
Packit 6c0a39
    const char *longname, sftp_attributes attr) {
Packit 6c0a39
  ssh_string name;
Packit 6c0a39
Packit 6c0a39
  name = ssh_string_from_char(file);
Packit 6c0a39
  if (name == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (msg->attrbuf == NULL) {
Packit 6c0a39
    msg->attrbuf = ssh_buffer_new();
Packit 6c0a39
    if (msg->attrbuf == NULL) {
Packit 6c0a39
      ssh_string_free(name);
Packit 6c0a39
      return -1;
Packit 6c0a39
    }
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_ssh_string(msg->attrbuf, name) < 0) {
Packit 6c0a39
    ssh_string_free(name);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  ssh_string_free(name);
Packit 6c0a39
  name = ssh_string_from_char(longname);
Packit 6c0a39
  if (name == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
  if (ssh_buffer_add_ssh_string(msg->attrbuf,name) < 0 ||
Packit 6c0a39
      buffer_add_attributes(msg->attrbuf,attr) < 0) {
Packit 6c0a39
    ssh_string_free(name);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
  ssh_string_free(name);
Packit 6c0a39
  msg->attr_num++;
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_names(sftp_client_message msg) {
Packit 6c0a39
  ssh_buffer out;
Packit 6c0a39
Packit 6c0a39
  out = ssh_buffer_new();
Packit 6c0a39
  if (out == NULL) {
Packit 6c0a39
    ssh_buffer_free(msg->attrbuf);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_u32(out, msg->id) < 0 ||
Packit 6c0a39
      ssh_buffer_add_u32(out, htonl(msg->attr_num)) < 0 ||
Packit 6c0a39
      ssh_buffer_add_data(out, ssh_buffer_get(msg->attrbuf),
Packit 6c0a39
        ssh_buffer_get_len(msg->attrbuf)) < 0 ||
Packit 6c0a39
      sftp_packet_write(msg->sftp, SSH_FXP_NAME, out) < 0) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    ssh_buffer_free(msg->attrbuf);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  ssh_buffer_free(out);
Packit 6c0a39
  ssh_buffer_free(msg->attrbuf);
Packit 6c0a39
Packit 6c0a39
  msg->attr_num = 0;
Packit 6c0a39
  msg->attrbuf = NULL;
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_status(sftp_client_message msg, uint32_t status,
Packit 6c0a39
    const char *message) {
Packit 6c0a39
  ssh_buffer out;
Packit 6c0a39
  ssh_string s;
Packit 6c0a39
Packit 6c0a39
  out = ssh_buffer_new();
Packit 6c0a39
  if (out == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  s = ssh_string_from_char(message ? message : "");
Packit 6c0a39
  if (s == NULL) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_u32(out, msg->id) < 0 ||
Packit 6c0a39
      ssh_buffer_add_u32(out, htonl(status)) < 0 ||
Packit 6c0a39
      ssh_buffer_add_ssh_string(out, s) < 0 ||
Packit 6c0a39
      ssh_buffer_add_u32(out, 0) < 0 || /* language string */
Packit 6c0a39
      sftp_packet_write(msg->sftp, SSH_FXP_STATUS, out) < 0) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    ssh_string_free(s);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  ssh_buffer_free(out);
Packit 6c0a39
  ssh_string_free(s);
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
int sftp_reply_data(sftp_client_message msg, const void *data, int len) {
Packit 6c0a39
  ssh_buffer out;
Packit 6c0a39
Packit 6c0a39
  out = ssh_buffer_new();
Packit 6c0a39
  if (out == NULL) {
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_buffer_add_u32(out, msg->id) < 0 ||
Packit 6c0a39
      ssh_buffer_add_u32(out, ntohl(len)) < 0 ||
Packit 6c0a39
      ssh_buffer_add_data(out, data, len) < 0 ||
Packit 6c0a39
      sftp_packet_write(msg->sftp, SSH_FXP_DATA, out) < 0) {
Packit 6c0a39
    ssh_buffer_free(out);
Packit 6c0a39
    return -1;
Packit 6c0a39
  }
Packit 6c0a39
  ssh_buffer_free(out);
Packit 6c0a39
Packit 6c0a39
  return 0;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
/*
Packit 6c0a39
 * This function will return you a new handle to give the client.
Packit 6c0a39
 * the function accepts an info that can be retrieved later with
Packit 6c0a39
 * the handle. Care is given that a corrupted handle won't give a
Packit 6c0a39
 * valid info (or worse).
Packit 6c0a39
 */
Packit 6c0a39
ssh_string sftp_handle_alloc(sftp_session sftp, void *info) {
Packit 6c0a39
  ssh_string ret;
Packit 6c0a39
  uint32_t val;
Packit 6c0a39
  int i;
Packit 6c0a39
Packit 6c0a39
  if (sftp->handles == NULL) {
Packit 6c0a39
    sftp->handles = calloc(SFTP_HANDLES, sizeof(void *));
Packit 6c0a39
    if (sftp->handles == NULL) {
Packit 6c0a39
      return NULL;
Packit 6c0a39
    }
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  for (i = 0; i < SFTP_HANDLES; i++) {
Packit 6c0a39
    if (sftp->handles[i] == NULL) {
Packit 6c0a39
      break;
Packit 6c0a39
    }
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (i == SFTP_HANDLES) {
Packit 6c0a39
    return NULL; /* no handle available */
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  val = i;
Packit 6c0a39
  ret = ssh_string_new(4);
Packit 6c0a39
  if (ret == NULL) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  memcpy(ssh_string_data(ret), &val, sizeof(uint32_t));
Packit 6c0a39
  sftp->handles[i] = info;
Packit 6c0a39
Packit 6c0a39
  return ret;
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
void *sftp_handle(sftp_session sftp, ssh_string handle){
Packit 6c0a39
  uint32_t val;
Packit 6c0a39
Packit 6c0a39
  if (sftp->handles == NULL) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  if (ssh_string_len(handle) != sizeof(uint32_t)) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  memcpy(&val, ssh_string_data(handle), sizeof(uint32_t));
Packit 6c0a39
Packit 6c0a39
  if (val > SFTP_HANDLES) {
Packit 6c0a39
    return NULL;
Packit 6c0a39
  }
Packit 6c0a39
Packit 6c0a39
  return sftp->handles[val];
Packit 6c0a39
}
Packit 6c0a39
Packit 6c0a39
void sftp_handle_remove(sftp_session sftp, void *handle) {
Packit 6c0a39
  int i;
Packit 6c0a39
Packit 6c0a39
  for (i = 0; i < SFTP_HANDLES; i++) {
Packit 6c0a39
    if (sftp->handles[i] == handle) {
Packit 6c0a39
      sftp->handles[i] = NULL;
Packit 6c0a39
      break;
Packit 6c0a39
    }
Packit 6c0a39
  }
Packit 6c0a39
}