⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 blob.c

📁 samba最新软件
💻 C
📖 第 1 页 / 共 2 页
字号:
/*    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 + -