Files
QNX/lib/c/dispatch/name.c
2025-08-20 19:02:58 +08:00

203 lines
5.4 KiB
C

/*
* $QNXLicenseC:
* Copyright 2007, QNX Software Systems. All Rights Reserved.
*
* You must obtain a written license from and pay applicable license fees to QNX
* Software Systems before you may reproduce, modify or distribute this software,
* or any work that includes all or part of this software. Free development
* licenses are available for evaluation and non-commercial purposes. For more
* information visit http://licensing.qnx.com or email licensing@qnx.com.
*
* This file may contain contributions from others. Please review this entire
* file for other proprietary rights or license notices, as well as the QNX
* Development Suite License Guide at http://licensing.qnx.com/license-guide/
* for other information.
* $
*/
#include <errno.h>
#include <malloc.h>
#include <stddef.h>
#include <stdlib.h>
#include <share.h>
#include <fcntl.h>
#include <process.h>
#include <sys/iomsg.h>
#include <sys/resmgr.h>
#include <sys/pathmsg.h>
#include <sys/dispatch.h>
#include <sys/netmgr.h>
#include <string.h>
#include <stdio.h>
#include "dispatch.h"
#include "resmgr.h"
const char *__name_prefix_global = "/dev/name/global/";
const char *__name_prefix_local = "/dev/name/local/";
name_attach_t *name_attach(dispatch_t *dpp, const char *path, unsigned flags) {
name_attach_t *attach;
int auto_create, mntid, chid;
char *newname;
//Reserve leading /'s for future use
if (!path || !*path || *path == '/') {
errno = EINVAL;
return NULL;
}
//Don't allow ../ or .. attaches to escape
if ((newname = (char *)strstr(path, "..")) &&
(newname[2] == '/' || newname[2] == '\0')) {
errno = EINVAL;
return NULL;
}
newname = NULL;
mntid = chid = -1;
auto_create = (dpp) ? 0 : 1;
if (!dpp && !(dpp = dispatch_create())) {
errno = EINVAL;
return NULL;
}
if (flags & NAME_FLAG_ATTACH_GLOBAL) {
if (!(newname = alloca(strlen(__name_prefix_global) + strlen(path) + 1))) {
errno = ENOMEM;
goto failure;
}
strcpy(newname, __name_prefix_global);
}
else {
if (!(newname = alloca(strlen(__name_prefix_local) + strlen(path) + 1))) {
errno = ENOMEM;
goto failure;
}
strcpy(newname, __name_prefix_local);
}
strcat(newname, path);
//TODO: Make the last thing really an attribute matching us?
//TODO: Allow the user to specify a function they want to be called
/* We have to regist with global service manager, this is done by:
* 1) do an empty resmgr_attach() to setup things (in case dpp->chid isn't there)
* 2) send an _IO_CONNECT_OPEN, _IO_CONNECT_EXTRA_RESMGR_LINK, to the
* manager. (What we really want is _IO_CONENCT_LINK plus
* _IO_CONNECT_EXTRA_RESMGR_LINK, but PathMgr won't pass it down to GNS
* 3) if 2) faild, there is no local GNS exist, then ONLY IF the attach
* if for _NAME_FLAG_ATTACH_LOCAL, we will send a _IO_CONNECT_LINK,
* _IO_CONNECT_EXTRA_RESMGR_LINK, as if pathmgr_link() did, this will
* regist it with pathmgr. And everything still works in local nodes.
*/
{
unsigned nflag;
struct link *linkl;
struct _io_resmgr_link_extra extra;
/* make sure no real attach will happen */
nflag = flags & ~_RESMGR_FLAG_FTYPEONLY;
if (resmgr_attach(dpp, NULL, NULL, 0, nflag, NULL, NULL, NULL) == -1) {
goto failure;
}
if (!(linkl = _resmgr_link_alloc())) {
goto failure;
}
linkl->connect_funcs = 0;
linkl->io_funcs = 0;
linkl->handle = 0;
extra.nd = ND_LOCAL_NODE;
extra.pid = getpid();
extra.chid = dpp->chid;
extra.handle = mntid = linkl->id;
extra.file_type = _FTYPE_NAME;
extra.flags = flags & _RESMGR_FLAG_MASK;
linkl->link_id = _connect(PATHMGR_COID, newname, 0, 0,
SH_DENYNO, _IO_CONNECT_OPEN, 0, 0, _FTYPE_NAME,
_IO_CONNECT_EXTRA_RESMGR_LINK, sizeof extra, &extra,
0, 0, 0);
if (linkl->link_id == -1 && !(flags & NAME_FLAG_ATTACH_GLOBAL) &&
(errno == ENOTSUP || errno == ENOENT || errno == ENOSYS))
{
linkl->link_id = _connect(PATHMGR_COID, newname, 0, 0,
SH_DENYNO, _IO_CONNECT_LINK, 0, 0, _FTYPE_NAME,
_IO_CONNECT_EXTRA_RESMGR_LINK, sizeof extra, &extra,
0, 0, 0);
}
if (linkl->link_id == -1) {
_resmgr_link_free(linkl->id, _RESMGR_DETACH_ALL);
mntid = -1;
goto failure;
}
linkl->flags &= ~_RESMGR_LINK_HALFOPEN;
}
chid = _DPP(dpp)->chid;
if (!(attach = malloc(sizeof(*attach)))) {
errno = ENOMEM;
goto failure;
}
attach->dpp = dpp;
attach->chid = chid;
attach->mntid = mntid;
return attach;
failure:
if (mntid > 0) {
(void)resmgr_detach(dpp, mntid, 0);
}
if (auto_create && dpp) {
(void)dispatch_destroy(dpp);
}
return NULL;
}
/*
Most of the time w/ these function the user wants the
dpp to be autocreated, so the default is to destroy
it when we detach.
*/
int name_detach(name_attach_t *attach, unsigned flags) {
int ret = EOK;
if (attach) {
ret = resmgr_detach(attach->dpp, attach->mntid, 0);
if (!(flags & NAME_FLAG_DETACH_SAVEDPP)) {
(void)dispatch_destroy(attach->dpp);
}
free(attach);
}
return ret;
}
#if 0
/* TF NOTE: Can we do away with these all together? */
resmgr_context_t *name_context_alloc(dispatch_t *dpp) {
return resmgr_context_alloc(dpp);
}
void name_context_free(resmgr_context_t *ctp) {
resmgr_context_free(ctp);
}
resmgr_context_t *name_block(resmgr_context_t *ctp) {
return resmgr_block(ctp);
}
int name_handler(resmgr_context_t *ctp) {
_resmgr_handler(ctp);
return 0;
}
#endif
__SRCVERSION("name.c $Rev: 153052 $");