|
|
80f43a |
/***********************************************************/
|
|
|
80f43a |
/* ntux: native translation und extension */
|
|
|
80f43a |
/* Copyright (C) 2016--2018 Z. Gilboa */
|
|
|
80f43a |
/* Released under GPLv2 and GPLv3; see COPYING.NTUX. */
|
|
|
80f43a |
/***********************************************************/
|
|
|
80f43a |
|
|
|
80f43a |
#include <psxabi/sys_sysapi.h>
|
|
|
80f43a |
#include <psxabi/sys_stat.h>
|
|
|
80f43a |
#include <psxabi/sys_errno.h>
|
|
|
80f43a |
|
|
|
471bb0 |
#include <psxxfi/xfi_base.h>
|
|
|
471bb0 |
#include <psxxfi/xfi_acl.h>
|
|
|
eb0350 |
#include <psxxfi/xfi_fs.h>
|
|
|
80f43a |
#include <psxxfi/xfi_ofd.h>
|
|
|
80f43a |
#include <psxxfi/xfi_unicode.h>
|
|
|
80f43a |
|
|
|
471bb0 |
#include <ntapi/nt_acl.h>
|
|
|
471bb0 |
#include <ntapi/nt_file.h>
|
|
|
471bb0 |
|
|
|
80f43a |
#include <ntux/ntux.h>
|
|
|
80f43a |
#include "ntux_driver_impl.h"
|
|
|
80f43a |
#include "ntux_nolibc_impl.h"
|
|
|
80f43a |
#include "ntux_errinfo_impl.h"
|
|
|
80f43a |
|
|
|
eb0350 |
static int ntux_cmd_chmod_ret(int fd, struct __ofd * ofd, void * hasync, int ret)
|
|
|
eb0350 |
{
|
|
|
eb0350 |
if (hasync)
|
|
|
471bb0 |
__xfi_close_handle(hasync);
|
|
|
eb0350 |
|
|
|
eb0350 |
if (ofd)
|
|
|
eb0350 |
__xfi_ofd_ref_dec(ofd);
|
|
|
eb0350 |
|
|
|
eb0350 |
if (fd >= 0)
|
|
|
eb0350 |
__sys_close(fd);
|
|
|
eb0350 |
|
|
|
eb0350 |
return ret;
|
|
|
eb0350 |
}
|
|
|
eb0350 |
|
|
|
80f43a |
int ntux_cmd_chmod(const struct ntux_driver_ctx * dctx, const char * dunit)
|
|
|
80f43a |
{
|
|
|
eb0350 |
intptr_t ret;
|
|
|
eb0350 |
int32_t status;
|
|
|
eb0350 |
int fdout;
|
|
|
eb0350 |
int fdcwd;
|
|
|
eb0350 |
const unsigned char * unit;
|
|
|
eb0350 |
nt_sd * srcsd;
|
|
|
eb0350 |
nt_sd_common_buffer dstsd;
|
|
|
eb0350 |
nt_sd_common_meta meta;
|
|
|
eb0350 |
uint32_t access_owner;
|
|
|
eb0350 |
uint32_t access_group;
|
|
|
eb0350 |
uint32_t access_other;
|
|
|
eb0350 |
uint32_t access_admin;
|
|
|
0a741d |
uint32_t ace_flags;
|
|
|
eb0350 |
size_t size;
|
|
|
eb0350 |
int fd = -1;
|
|
|
eb0350 |
struct __ofd * ofd = 0;
|
|
|
eb0350 |
void * hasync = 0;
|
|
|
eb0350 |
uint32_t buf[0x300];
|
|
|
eb0350 |
|
|
|
0a741d |
/* initial version: --strmode only */
|
|
|
0a741d |
if (!dctx->cctx->strmode)
|
|
|
0a741d |
return ntux_cmd_chmod_ret(
|
|
|
0a741d |
0,0,0,
|
|
|
0a741d |
NTUX_CUSTOM_ERROR(
|
|
|
0a741d |
dctx,
|
|
|
0a741d |
NTUX_ERR_FLEE_ERROR));
|
|
|
0a741d |
|
|
|
0a741d |
/* ACE propagation: +p, -p */
|
|
|
0a741d |
if (!strcmp(dctx->cctx->strmode,"+p"))
|
|
|
0a741d |
ace_flags = NT_ACE_CONTAINER_INHERIT | NT_ACE_OBJECT_INHERIT;
|
|
|
0a741d |
|
|
|
0a741d |
else if (!strcmp(dctx->cctx->strmode,"-p"))
|
|
|
0a741d |
ace_flags = 0;
|
|
|
0a741d |
|
|
|
0a741d |
else
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
0,0,0,
|
|
|
eb0350 |
NTUX_CUSTOM_ERROR(
|
|
|
eb0350 |
dctx,
|
|
|
eb0350 |
NTUX_ERR_FLEE_ERROR));
|
|
|
eb0350 |
|
|
|
eb0350 |
/* init */
|
|
|
eb0350 |
ntux_driver_set_ectx(
|
|
|
eb0350 |
dctx,0,dunit);
|
|
|
eb0350 |
|
|
|
eb0350 |
unit = (const unsigned char *)dunit;
|
|
|
eb0350 |
|
|
|
eb0350 |
/* fdctx */
|
|
|
eb0350 |
fdout = ntux_driver_fdout(dctx);
|
|
|
eb0350 |
fdcwd = ntux_driver_fdcwd(dctx);
|
|
|
eb0350 |
|
|
|
eb0350 |
/* fd */
|
|
|
eb0350 |
if ((ret = __sys_openat(fdcwd,unit,0,0)) < 0)
|
|
|
eb0350 |
if (ntux_errno_set(dctx,ret))
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
0,0,0,
|
|
|
eb0350 |
NTUX_SYSTEM_ERROR(dctx));
|
|
|
eb0350 |
|
|
|
eb0350 |
fd = ret;
|
|
|
eb0350 |
|
|
|
eb0350 |
/* ofd */
|
|
|
eb0350 |
if (!(ofd = __xfi_ofd_ref_inc(fd)))
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
fd,0,0,
|
|
|
eb0350 |
NTUX_CUSTOM_ERROR(
|
|
|
eb0350 |
dctx,
|
|
|
eb0350 |
NTUX_ERR_FLOW_ERROR));
|
|
|
eb0350 |
|
|
|
eb0350 |
/* hasync */
|
|
|
eb0350 |
if ((status = __xfi_fs_open_async(
|
|
|
eb0350 |
&hasync,
|
|
|
eb0350 |
ofd->info.hfile,0,
|
|
|
eb0350 |
NT_SEC_READ_CONTROL
|
|
|
eb0350 |
| NT_SEC_WRITE_DAC,
|
|
|
eb0350 |
NT_FILE_SHARE_READ
|
|
|
eb0350 |
| NT_FILE_SHARE_WRITE
|
|
|
eb0350 |
| NT_FILE_SHARE_DELETE)))
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
fd,ofd,0,
|
|
|
eb0350 |
NTUX_SYSTEM_ERROR(dctx));
|
|
|
eb0350 |
|
|
|
eb0350 |
/* srcsd */
|
|
|
eb0350 |
srcsd = (nt_sd *)buf;
|
|
|
eb0350 |
|
|
|
471bb0 |
if ((status = __xfi_query_security_object(
|
|
|
eb0350 |
hasync,
|
|
|
eb0350 |
NT_OWNER_SECURITY_INFORMATION
|
|
|
eb0350 |
| NT_GROUP_SECURITY_INFORMATION
|
|
|
eb0350 |
| NT_DACL_SECURITY_INFORMATION,
|
|
|
eb0350 |
srcsd,sizeof(buf),&size)))
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
fd,ofd,hasync,
|
|
|
eb0350 |
NTUX_SYSTEM_ERROR(dctx));
|
|
|
eb0350 |
|
|
|
471bb0 |
if ((status = __xfi_acl_init_common_descriptor_meta(
|
|
|
eb0350 |
&meta,srcsd,
|
|
|
eb0350 |
NT_ACL_INIT_COMMON_DESCRIPTION_META_STRICT_MODE)))
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
fd,ofd,hasync,
|
|
|
eb0350 |
NTUX_SYSTEM_ERROR(dctx));
|
|
|
eb0350 |
|
|
|
eb0350 |
/* source permissions */
|
|
|
eb0350 |
access_owner = meta.owner_ace ? meta.owner_ace->mask : 0;
|
|
|
eb0350 |
access_group = meta.group_ace ? meta.group_ace->mask : 0;
|
|
|
eb0350 |
access_other = meta.other_ace ? meta.other_ace->mask : 0;
|
|
|
0a741d |
access_admin = meta.admin_ace ? meta.admin_ace->mask : 0;
|
|
|
eb0350 |
|
|
|
eb0350 |
/* updated dacl */
|
|
|
471bb0 |
__xfi_acl_init_common_descriptor(
|
|
|
eb0350 |
&dstsd,
|
|
|
eb0350 |
meta.owner,meta.group,0,0,
|
|
|
eb0350 |
access_owner,access_group,access_other,
|
|
|
0a741d |
access_admin,meta.system_acc,
|
|
|
0a741d |
ace_flags);
|
|
|
eb0350 |
|
|
|
471bb0 |
if ((status = __xfi_set_security_object(
|
|
|
eb0350 |
hasync,
|
|
|
eb0350 |
NT_DACL_SECURITY_INFORMATION,
|
|
|
eb0350 |
&dstsd.sd)))
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(
|
|
|
eb0350 |
fd,ofd,hasync,
|
|
|
eb0350 |
NTUX_SYSTEM_ERROR(dctx));
|
|
|
eb0350 |
|
|
|
eb0350 |
/* changes */
|
|
|
eb0350 |
(void)fdout;
|
|
|
80f43a |
|
|
|
eb0350 |
/* all done */
|
|
|
eb0350 |
return ntux_cmd_chmod_ret(fd,ofd,hasync,0);
|
|
|
80f43a |
}
|