📄 blob.c
字号:
/* Unix SMB/CIFS implementation. Copyright (C) Andrew Tridgell 2003 Copyright (C) Stefan Metzmacher 2006 This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>.*/#include "includes.h"#include "lib/util/dlinklist.h"#include "smb_server/smb_server.h"#include "librpc/gen_ndr/ndr_misc.h"#include "ntvfs/ntvfs.h"#include "libcli/raw/libcliraw.h"#include "libcli/raw/raw_proto.h"#include "param/param.h"#define BLOB_CHECK(cmd) do { \ NTSTATUS _status; \ _status = cmd; \ NT_STATUS_NOT_OK_RETURN(_status); \} while (0)#define BLOB_CHECK_MIN_SIZE(blob, size) do { \ if ((blob)->length < (size)) { \ return NT_STATUS_INFO_LENGTH_MISMATCH; \ } \} while (0)/* align the end of the blob on an 8 byte boundary */#define BLOB_ALIGN(blob, alignment) do { \ if ((blob)->length & ((alignment)-1)) { \ uint8_t _pad = (alignment) - ((blob)->length & ((alignment)-1)); \ BLOB_CHECK(smbsrv_blob_fill_data(blob, blob, (blob)->length+_pad)); \ } \} while (0)/* grow the data size of a trans2 reply */NTSTATUS smbsrv_blob_grow_data(TALLOC_CTX *mem_ctx, DATA_BLOB *blob, uint32_t new_size){ if (new_size > blob->length) { uint8_t *p; p = talloc_realloc(mem_ctx, blob->data, uint8_t, new_size); NT_STATUS_HAVE_NO_MEMORY(p); blob->data = p; } blob->length = new_size; return NT_STATUS_OK;}/* grow the data, zero filling any new bytes */NTSTATUS smbsrv_blob_fill_data(TALLOC_CTX *mem_ctx, DATA_BLOB *blob, uint32_t new_size){ uint32_t old_size = blob->length; BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, new_size)); if (new_size > old_size) { memset(blob->data + old_size, 0, new_size - old_size); } return NT_STATUS_OK;}/* pull a string from a blob in a trans2 request*/size_t smbsrv_blob_pull_string(struct request_bufinfo *bufinfo, const DATA_BLOB *blob, uint16_t offset, const char **str, int flags){ *str = NULL; /* we use STR_NO_RANGE_CHECK because the params are allocated separately in a DATA_BLOB, so we need to do our own range checking */ if (offset >= blob->length) { return 0; } return req_pull_string(bufinfo, str, blob->data + offset, blob->length - offset, STR_NO_RANGE_CHECK | flags);}/* push a string into the data section of a trans2 request return the number of bytes consumed in the output*/size_t smbsrv_blob_push_string(TALLOC_CTX *mem_ctx, DATA_BLOB *blob, uint32_t len_offset, uint32_t offset, const char *str, int dest_len, int default_flags, int flags){ int alignment = 0, ret = 0, pkt_len; /* we use STR_NO_RANGE_CHECK because the params are allocated separately in a DATA_BLOB, so we need to do our own range checking */ if (!str || offset >= blob->length) { if (flags & STR_LEN8BIT) { SCVAL(blob->data, len_offset, 0); } else { SIVAL(blob->data, len_offset, 0); } return 0; } flags |= STR_NO_RANGE_CHECK; if (dest_len == -1 || (dest_len > blob->length - offset)) { dest_len = blob->length - offset; } if (!(flags & (STR_ASCII|STR_UNICODE))) { flags |= default_flags; } if ((offset&1) && (flags & STR_UNICODE) && !(flags & STR_NOALIGN)) { alignment = 1; if (dest_len > 0) { SCVAL(blob->data + offset, 0, 0); ret = push_string(lp_iconv_convenience(global_loadparm), blob->data + offset + 1, str, dest_len-1, flags); } } else { ret = push_string(lp_iconv_convenience(global_loadparm), blob->data + offset, str, dest_len, flags); } /* sometimes the string needs to be terminated, but the length on the wire must not include the termination! */ pkt_len = ret; if ((flags & STR_LEN_NOTERM) && (flags & STR_TERMINATE)) { if ((flags & STR_UNICODE) && ret >= 2) { pkt_len = ret-2; } if ((flags & STR_ASCII) && ret >= 1) { pkt_len = ret-1; } } if (flags & STR_LEN8BIT) { SCVAL(blob->data, len_offset, pkt_len); } else { SIVAL(blob->data, len_offset, pkt_len); } return ret + alignment;}/* append a string to the data section of a trans2 reply len_offset points to the place in the packet where the length field should go*/NTSTATUS smbsrv_blob_append_string(TALLOC_CTX *mem_ctx, DATA_BLOB *blob, const char *str, uint_t len_offset, int default_flags, int flags){ size_t ret; uint32_t offset; const int max_bytes_per_char = 3; offset = blob->length; BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, offset + (2+strlen_m(str))*max_bytes_per_char)); ret = smbsrv_blob_push_string(mem_ctx, blob, len_offset, offset, str, -1, default_flags, flags); if (ret < 0) { return NT_STATUS_FOOBAR; } BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, offset + ret)); return NT_STATUS_OK;}NTSTATUS smbsrv_push_passthru_fsinfo(TALLOC_CTX *mem_ctx, DATA_BLOB *blob, enum smb_fsinfo_level level, union smb_fsinfo *fsinfo, int default_str_flags){ uint_t i; DATA_BLOB guid_blob; switch (level) { case RAW_QFS_VOLUME_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 18)); push_nttime(blob->data, 0, fsinfo->volume_info.out.create_time); SIVAL(blob->data, 8, fsinfo->volume_info.out.serial_number); SSVAL(blob->data, 16, 0); /* padding */ BLOB_CHECK(smbsrv_blob_append_string(mem_ctx, blob, fsinfo->volume_info.out.volume_name.s, 12, default_str_flags, STR_UNICODE)); return NT_STATUS_OK; case RAW_QFS_SIZE_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 24)); SBVAL(blob->data, 0, fsinfo->size_info.out.total_alloc_units); SBVAL(blob->data, 8, fsinfo->size_info.out.avail_alloc_units); SIVAL(blob->data, 16, fsinfo->size_info.out.sectors_per_unit); SIVAL(blob->data, 20, fsinfo->size_info.out.bytes_per_sector); return NT_STATUS_OK; case RAW_QFS_DEVICE_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 8)); SIVAL(blob->data, 0, fsinfo->device_info.out.device_type); SIVAL(blob->data, 4, fsinfo->device_info.out.characteristics); return NT_STATUS_OK; case RAW_QFS_ATTRIBUTE_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 12)); SIVAL(blob->data, 0, fsinfo->attribute_info.out.fs_attr); SIVAL(blob->data, 4, fsinfo->attribute_info.out.max_file_component_length); /* this must not be null terminated or win98 gets confused! also note that w2k3 returns this as unicode even when ascii is negotiated */ BLOB_CHECK(smbsrv_blob_append_string(mem_ctx, blob, fsinfo->attribute_info.out.fs_type.s, 8, default_str_flags, STR_UNICODE)); return NT_STATUS_OK; case RAW_QFS_QUOTA_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 48)); SBVAL(blob->data, 0, fsinfo->quota_information.out.unknown[0]); SBVAL(blob->data, 8, fsinfo->quota_information.out.unknown[1]); SBVAL(blob->data, 16, fsinfo->quota_information.out.unknown[2]); SBVAL(blob->data, 24, fsinfo->quota_information.out.quota_soft); SBVAL(blob->data, 32, fsinfo->quota_information.out.quota_hard); SBVAL(blob->data, 40, fsinfo->quota_information.out.quota_flags); return NT_STATUS_OK; case RAW_QFS_FULL_SIZE_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 32)); SBVAL(blob->data, 0, fsinfo->full_size_information.out.total_alloc_units); SBVAL(blob->data, 8, fsinfo->full_size_information.out.call_avail_alloc_units); SBVAL(blob->data, 16, fsinfo->full_size_information.out.actual_avail_alloc_units); SIVAL(blob->data, 24, fsinfo->full_size_information.out.sectors_per_unit); SIVAL(blob->data, 28, fsinfo->full_size_information.out.bytes_per_sector); return NT_STATUS_OK; case RAW_QFS_OBJECTID_INFORMATION: { enum ndr_err_code ndr_err; BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 64)); ndr_err = ndr_push_struct_blob(&guid_blob, mem_ctx, NULL, &fsinfo->objectid_information.out.guid, (ndr_push_flags_fn_t)ndr_push_GUID); if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) { BLOB_CHECK(ndr_map_error2ntstatus(ndr_err)); } memcpy(blob->data, guid_blob.data, guid_blob.length); for (i=0;i<6;i++) { SBVAL(blob->data, 16 + 8*i, fsinfo->objectid_information.out.unknown[i]); } return NT_STATUS_OK; } default: return NT_STATUS_INVALID_LEVEL; } return NT_STATUS_INVALID_LEVEL;}NTSTATUS smbsrv_push_passthru_fileinfo(TALLOC_CTX *mem_ctx, DATA_BLOB *blob, enum smb_fileinfo_level level, union smb_fileinfo *st, int default_str_flags){ uint_t i; size_t list_size; switch (level) { case RAW_FILEINFO_BASIC_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 40)); push_nttime(blob->data, 0, st->basic_info.out.create_time); push_nttime(blob->data, 8, st->basic_info.out.access_time); push_nttime(blob->data, 16, st->basic_info.out.write_time); push_nttime(blob->data, 24, st->basic_info.out.change_time); SIVAL(blob->data, 32, st->basic_info.out.attrib); SIVAL(blob->data, 36, 0); /* padding */ return NT_STATUS_OK; case RAW_FILEINFO_NETWORK_OPEN_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 56)); push_nttime(blob->data, 0, st->network_open_information.out.create_time); push_nttime(blob->data, 8, st->network_open_information.out.access_time); push_nttime(blob->data, 16, st->network_open_information.out.write_time); push_nttime(blob->data, 24, st->network_open_information.out.change_time); SBVAL(blob->data, 32, st->network_open_information.out.alloc_size); SBVAL(blob->data, 40, st->network_open_information.out.size); SIVAL(blob->data, 48, st->network_open_information.out.attrib); SIVAL(blob->data, 52, 0); /* padding */ return NT_STATUS_OK; case RAW_FILEINFO_STANDARD_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 24)); SBVAL(blob->data, 0, st->standard_info.out.alloc_size); SBVAL(blob->data, 8, st->standard_info.out.size); SIVAL(blob->data, 16, st->standard_info.out.nlink); SCVAL(blob->data, 20, st->standard_info.out.delete_pending); SCVAL(blob->data, 21, st->standard_info.out.directory); SSVAL(blob->data, 22, 0); /* padding */ return NT_STATUS_OK; case RAW_FILEINFO_ATTRIBUTE_TAG_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 8)); SIVAL(blob->data, 0, st->attribute_tag_information.out.attrib); SIVAL(blob->data, 4, st->attribute_tag_information.out.reparse_tag); return NT_STATUS_OK; case RAW_FILEINFO_EA_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 4)); SIVAL(blob->data, 0, st->ea_info.out.ea_size); return NT_STATUS_OK; case RAW_FILEINFO_MODE_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 4)); SIVAL(blob->data, 0, st->mode_information.out.mode); return NT_STATUS_OK; case RAW_FILEINFO_ALIGNMENT_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 4)); SIVAL(blob->data, 0, st->alignment_information.out.alignment_requirement); return NT_STATUS_OK; case RAW_FILEINFO_ACCESS_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 4)); SIVAL(blob->data, 0, st->access_information.out.access_flags); return NT_STATUS_OK; case RAW_FILEINFO_POSITION_INFORMATION: BLOB_CHECK(smbsrv_blob_grow_data(mem_ctx, blob, 8)); SBVAL(blob->data, 0, st->position_information.out.position); return NT_STATUS_OK; case RAW_FILEINFO_COMPRESSION_INFORMATION:
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -