Blame src/msq/ntapi_msq_connect.c

47f21a
/********************************************************/
47f21a
/*  ntapi: Native API core library                      */
47f21a
/*  Copyright (C) 2013--2017  Z. Gilboa                 */
47f21a
/*  Released under GPLv2 and GPLv3; see COPYING.NTAPI.  */
47f21a
/********************************************************/
47f21a
47f21a
#include <psxtypes/psxtypes.h>
47f21a
#include <ntapi/nt_file.h>
47f21a
#include <ntapi/nt_string.h>
47f21a
#include <ntapi/nt_atomic.h>
47f21a
#include <ntapi/nt_port.h>
47f21a
#include <ntapi/nt_ipc.h>
47f21a
#include <ntapi/nt_msq.h>
47f21a
#include <ntapi/ntapi.h>
47f21a
#include "ntapi_impl.h"
47f21a
47f21a
static const nt_guid	g_msqpid    = NT_IPC_GUID_MSQPID;
47f21a
static const wchar16_t	p_msqpid[6] = NT_IPC_OBJDIR_PREFIX_MSQPID;
47f21a
47f21a
static int32_t __msqctl_get_service_attr(
47f21a
	nt_rtdata *		rtdata,
47f21a
	nt_tty_service_info *	msqctl)
47f21a
{
47f21a
	nt_iosb iosb;
47f21a
47f21a
	/* inherited runtime data? */
47f21a
	if (rtdata->msqctl_keys[0]) {
47f21a
		msqctl->attr.ver_major = 0;
47f21a
		msqctl->attr.ver_minor = 0;
47f21a
		msqctl->attr.options   = 0;
47f21a
		msqctl->attr.flags     = 0;
47f21a
47f21a
		msqctl->attr.type    = rtdata->msqctl_type;
47f21a
		msqctl->attr.subtype = rtdata->msqctl_subtype;
47f21a
47f21a
		msqctl->attr.keys.key[0] = rtdata->msqctl_keys[0];
47f21a
		msqctl->attr.keys.key[1] = rtdata->msqctl_keys[1];
47f21a
		msqctl->attr.keys.key[2] = rtdata->msqctl_keys[2];
47f21a
		msqctl->attr.keys.key[3] = rtdata->msqctl_keys[3];
47f21a
		msqctl->attr.keys.key[4] = rtdata->msqctl_keys[4];
47f21a
		msqctl->attr.keys.key[5] = rtdata->msqctl_keys[5];
47f21a
47f21a
		__ntapi->tt_guid_copy(
47f21a
			&msqctl->attr.guid,
47f21a
			&rtdata->msqctl_guid);
47f21a
47f21a
		return NT_STATUS_SUCCESS;
47f21a
	}
47f21a
47f21a
	/* obtain service info */
47f21a
	return __ntapi->tty_query_information_service(
47f21a
		0,&iosb,msqctl,
47f21a
		&(nt_guid)NT_PORT_GUID_MSQCTL,
47f21a
		0,0);
47f21a
}
47f21a
47f21a
static int32_t __msqctl_server_connect(
47f21a
	nt_rtdata *		rtdata,
47f21a
	nt_tty_service_info *	msqctl)
47f21a
{
47f21a
	int32_t status;
47f21a
47f21a
	/* already cononected? */
47f21a
	if (rtdata->hmsqctl)
47f21a
		return NT_STATUS_SUCCESS;
47f21a
47f21a
	/* connect */
47f21a
	if ((status = __ntapi->ipc_connect_by_attr(
47f21a
			&rtdata->hmsqctl,
47f21a
			&msqctl->attr)))
47f21a
		return status;
47f21a
47f21a
	/* update */
47f21a
	rtdata->msqctl_type    = msqctl->attr.type;
47f21a
	rtdata->msqctl_subtype = msqctl->attr.subtype;
47f21a
47f21a
	rtdata->msqctl_keys[0] = msqctl->attr.keys.key[0];
47f21a
	rtdata->msqctl_keys[1] = msqctl->attr.keys.key[1];
47f21a
	rtdata->msqctl_keys[2] = msqctl->attr.keys.key[2];
47f21a
	rtdata->msqctl_keys[3] = msqctl->attr.keys.key[3];
47f21a
	rtdata->msqctl_keys[4] = msqctl->attr.keys.key[4];
47f21a
	rtdata->msqctl_keys[5] = msqctl->attr.keys.key[5];
47f21a
47f21a
	__ntapi->tt_guid_copy(
47f21a
		&rtdata->msqctl_guid,
47f21a
		&msqctl->attr.guid);
47f21a
47f21a
	return NT_STATUS_SUCCESS;
47f21a
}
47f21a
47f21a
static int32_t __msqpid_symlink_set(
47f21a
	nt_rtdata *		rtdata,
47f21a
	nt_tty_service_info *	msqctl)
47f21a
{
47f21a
	int32_t			status;
47f21a
	void *			hpiddir;
47f21a
	nt_port_name		svcname;
47f21a
	nt_unicode_string	str;
47f21a
47f21a
	if (rtdata->hmsqpid)
47f21a
		return NT_STATUS_SUCCESS;
47f21a
47f21a
	if (!rtdata->hmsqpiddir) {
47f21a
		if ((status = __ntapi->tt_open_ipc_object_directory(
47f21a
				&hpiddir,
47f21a
				NT_DIRECTORY_ALL_ACCESS,
47f21a
				p_msqpid,&g_msqpid)))
47f21a
			return status;
47f21a
47f21a
		if (at_locked_cas((intptr_t *)&rtdata->hmsqpiddir,0,(intptr_t)hpiddir))
47f21a
			__ntapi->zw_close(hpiddir);
47f21a
	}
47f21a
47f21a
	__ntapi->tt_port_name_from_attr(
47f21a
		&svcname,&msqctl->attr);
47f21a
47f21a
	str.strlen = (uint16_t)(size_t)(&(((nt_port_name *)0)->null_termination));
47f21a
	str.maxlen = sizeof(nt_port_name);
47f21a
	str.buffer = svcname.base_named_objects;
47f21a
47f21a
	return __ntapi->tt_create_ipc_object_directory_entry(
47f21a
		&rtdata->hmsqpid,
47f21a
		NT_SEC_STANDARD_RIGHTS_ALL,
47f21a
		rtdata->hmsqpiddir,
47f21a
		0,&str,
47f21a
		pe_get_current_process_id());
47f21a
47f21a
}
47f21a
47f21a
static int32_t	__stdcall __msq_open(
47f21a
	void *			hipc,
47f21a
	nt_msq_info *		msq,
47f21a
	uint32_t		access,
47f21a
	nt_object_attributes *	oa,
47f21a
	nt_iosb *		iosb,
47f21a
	uint32_t		share,
47f21a
	uint32_t		msqslots,
47f21a
	uint32_t		key,
47f21a
	uint32_t		id,
47f21a
	uint32_t		opcode)
47f21a
{
47f21a
	int32_t			status;
47f21a
	nt_tty_port_msg		msg;
47f21a
	nt_iosb			siosb;
47f21a
	nt_tty_service_info	msqctl;
47f21a
	nt_runtime_data *	rtdata;
47f21a
47f21a
	/* init */
47f21a
	rtdata = (__ntapi_internals())->rtdata;
47f21a
47f21a
	/* msqctl service attributes */
47f21a
	if (!rtdata->hmsqpid)
47f21a
		if ((status = __msqctl_get_service_attr(rtdata,&msqctl)))
47f21a
			return status;
47f21a
47f21a
	/* msqctl server */
47f21a
	if ((status = __msqctl_server_connect(rtdata,&msqctl)))
47f21a
		return status;
47f21a
47f21a
	/* msqpid symlink */
47f21a
	if ((status = __msqpid_symlink_set(rtdata,&msqctl)))
47f21a
		return status;
47f21a
47f21a
	/* hipc */
47f21a
	if (!hipc && (opcode == NT_TTY_MSQ_ALLOC))
47f21a
		hipc = (__ntapi_internals())->rtdata->hmsqctl;
47f21a
47f21a
	/* obtain msq info */
47f21a
	__ntapi->tt_aligned_block_memset(
47f21a
		&msg,0,sizeof(msg));
47f21a
47f21a
	if (!iosb)
47f21a
		iosb = &siosb;
47f21a
47f21a
	msg.header.msg_type		= NT_LPC_NEW_MESSAGE;
47f21a
	msg.header.data_size		= sizeof(nt_msq_info_msg) - sizeof(msg.header);
47f21a
	msg.header.msg_size		= sizeof(msg);
47f21a
	msg.ttyinfo.opcode		= opcode;
47f21a
47f21a
	msg.msqinfo.msqkey		= (int32_t)key;
47f21a
	msg.msqinfo.msqid		= (int32_t)id;
47f21a
	msg.msqinfo.msqslots		= msqslots;
47f21a
47f21a
	msg.msqinfo.ntattr		= oa->obj_attr;
47f21a
	msg.msqinfo.ntaccess		= access;
47f21a
	msg.msqinfo.ntshare		= share;
47f21a
47f21a
	if ((status = __ntapi->zw_request_wait_reply_port(hipc,&msg,&msg)))
47f21a
		return status;
47f21a
	else if (msg.ttyinfo.status)
47f21a
		return msg.ttyinfo.status;
47f21a
47f21a
	iosb->info   = sizeof(msg.svcinfo);
47f21a
	iosb->status = NT_STATUS_SUCCESS;
47f21a
47f21a
	/* new msqueue? */
47f21a
	if (opcode == NT_TTY_MSQ_ALLOC)
47f21a
		if ((status = __ntapi->ipc_connect_by_attr(
47f21a
				&hipc,&msg.svcinfo.attr)))
47f21a
			return status;
47f21a
47f21a
	/* all done */
47f21a
	__ntapi->tt_aligned_block_memset(
47f21a
		(uintptr_t *)msq,
47f21a
		0,sizeof(*msq));
47f21a
47f21a
	msq->msqkey = msg.svcinfo.key;
47f21a
	msq->msqid  = msg.svcinfo.id;
47f21a
47f21a
	msq->hport = hipc;
47f21a
47f21a
	return NT_STATUS_SUCCESS;
47f21a
}
47f21a
47f21a
47f21a
int32_t	__ntapi_msq_create(
47f21a
	__in	void *			hport,
47f21a
	__out	nt_msq_info *		msq,
47f21a
	__in	uint32_t		access,
47f21a
	__in	nt_object_attributes *	oa,
47f21a
	__out	nt_iosb *		iosb,
47f21a
	__in	uint32_t		share,
47f21a
	__in	uint32_t		nslots)
47f21a
{
47f21a
	uint32_t key;
47f21a
47f21a
	/* validate */
47f21a
	if (!oa->root_dir)
47f21a
		return NT_STATUS_DIRECTORY_SERVICE_REQUIRED;
47f21a
47f21a
	if (oa->obj_name && !oa->obj_name->strlen) {
47f21a
		key = 0;
47f21a
47f21a
	} else if (oa->obj_name) {
47f21a
		if (oa->obj_name->strlen != 8 * sizeof(wchar16_t))
47f21a
			return NT_STATUS_OBJECT_NAME_INVALID;
47f21a
47f21a
		if (__ntapi->tt_hex_utf16_to_uint32(oa->obj_name->buffer,&key))
47f21a
			return NT_STATUS_OBJECT_NAME_INVALID;
47f21a
47f21a
	} else {
47f21a
		key = 0;
47f21a
	}
47f21a
47f21a
	/* open msqueue */
47f21a
	return __msq_open(
47f21a
		hport,msq,access,
47f21a
		oa,iosb,share,nslots,
47f21a
		key,0,NT_TTY_MSQ_ALLOC);
47f21a
}
47f21a
47f21a
47f21a
int32_t	__stdcall __ntapi_msq_open(
47f21a
	__in	void *			hport,
47f21a
	__out	nt_msq_info *		msq,
47f21a
	__in	uint32_t		access,
47f21a
	__in	nt_object_attributes *	oa,
47f21a
	__out	nt_iosb *		iosb,
47f21a
	__in	uint32_t		share,
47f21a
	__in	uint32_t		nslots)
47f21a
{
47f21a
	int32_t			status;
47f21a
	uint32_t		key;
47f21a
	uint32_t		id;
47f21a
	void *			hsymlink;
47f21a
	nt_oa			ipcoa;
47f21a
	void *			hipc;
47f21a
	nt_rtdata *		rtdata;
47f21a
	nt_tty_service_info	msqctl;
47f21a
47f21a
	/* init */
47f21a
	rtdata = (__ntapi_internals())->rtdata;
47f21a
47f21a
	/* validate */
47f21a
	if (!oa->root_dir)
47f21a
		return NT_STATUS_DIRECTORY_SERVICE_REQUIRED;
47f21a
47f21a
	if (!oa->obj_name)
47f21a
		return NT_STATUS_INVALID_PARAMETER;
47f21a
47f21a
	if (oa->obj_name->strlen != 8 * sizeof(wchar16_t))
47f21a
		return NT_STATUS_OBJECT_NAME_INVALID;
47f21a
47f21a
	if (__ntapi->tt_hex_utf16_to_uint32(oa->obj_name->buffer,&key))
47f21a
		return NT_STATUS_OBJECT_NAME_INVALID;
47f21a
47f21a
	/* open symlink */
47f21a
	ipcoa.len      = sizeof(ipcoa);
47f21a
	ipcoa.root_dir = oa->root_dir;
47f21a
	ipcoa.obj_name = oa->obj_name;
47f21a
	ipcoa.obj_attr = 0;
47f21a
	ipcoa.sec_desc = oa->sec_desc;
47f21a
	ipcoa.sec_qos  = oa->sec_qos;
47f21a
47f21a
	status = __ntapi->zw_open_symbolic_link_object(
47f21a
		&hsymlink,
47f21a
		NT_SEC_STANDARD_RIGHTS_READ | NT_GENERIC_READ,
47f21a
		&ipcoa);
47f21a
47f21a
	switch (status) {
47f21a
		case NT_STATUS_SUCCESS:
47f21a
			break;
47f21a
47f21a
		case NT_STATUS_OBJECT_NAME_NOT_FOUND:
47f21a
		case NT_STATUS_OBJECT_PATH_NOT_FOUND:
47f21a
			if (oa->obj_attr & NT_OBJ_OPENIF)
47f21a
				return __msq_open(
47f21a
					hport,msq,access,
47f21a
					oa,iosb,share,nslots,
47f21a
					key,0,NT_TTY_MSQ_ALLOC);
47f21a
			else
47f21a
				return status;
47f21a
47f21a
		default:
47f21a
			return status;
47f21a
	}
47f21a
47f21a
	/* msqctl service attributes */
47f21a
	if (!rtdata->hmsqpid)
47f21a
		if ((status = __msqctl_get_service_attr(rtdata,&msqctl)))
47f21a
			return status;
47f21a
47f21a
	/* msqctl server */
47f21a
	if ((status = __msqctl_server_connect(rtdata,&msqctl)))
47f21a
		return status;
47f21a
47f21a
	/* msqpid symlink */
47f21a
	if ((status = __msqpid_symlink_set(rtdata,&msqctl)))
47f21a
		return status;
47f21a
47f21a
	/* ipc connect */
47f21a
	status = __ntapi->ipc_connect_by_symlink(
47f21a
		&hipc,hsymlink);
47f21a
47f21a
	__ntapi->zw_close(
47f21a
		hsymlink);
47f21a
47f21a
	if (status)
47f21a
		return status;
47f21a
47f21a
	/* open by id? */
47f21a
	if (oa->obj_attr & NT_OBJ_OPENLINK) {
47f21a
		id  = key;
47f21a
		key = 0;
47f21a
	} else {
47f21a
		id  = 0;
47f21a
	}
47f21a
47f21a
	return __msq_open(
47f21a
		hipc,msq,access,
47f21a
		oa,iosb,share,nslots,
47f21a
		key,id,NT_TTY_MSQ_OPEN);
47f21a
}