-
Notifications
You must be signed in to change notification settings - Fork 6
Expand file tree
/
Copy pathobject_id.c
More file actions
158 lines (137 loc) · 3.79 KB
/
object_id.c
File metadata and controls
158 lines (137 loc) · 3.79 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Pocessing of object ids
*
* Part of this file is based on code from the NTFS-3G.
*
* Copyright (c) 2009-2019 Jean-Pierre Andre
* Copyright (c) 2026 LG Electronics Co., Ltd.
*/
#include "ntfs.h"
#include "index.h"
#include "object_id.h"
struct object_id_index_key {
union {
u32 alignment;
struct guid guid;
} object_id;
} __packed;
struct object_id_index_data {
__le64 file_id;
struct guid birth_volume_id;
struct guid birth_object_id;
struct guid domain_id;
} __packed;
/* Index entry in $Extend/$ObjId */
struct object_id_index {
struct index_entry_header header;
struct object_id_index_key key;
struct object_id_index_data data;
} __packed;
__le16 objid_index_name[] = {cpu_to_le16('$'), cpu_to_le16('O'), 0};
/*
* open_object_id_index - Open the $Extend/$ObjId file and its index
* @vol: NTFS volume structure
*
* Opens the $ObjId system file and retrieves its index context.
*
* Return: The index context if opened successfully, or NULL if an error
* occurred.
*/
static struct ntfs_index_context *open_object_id_index(struct ntfs_volume *vol)
{
struct inode *dir_vi, *vi;
struct ntfs_inode *dir_ni;
struct ntfs_index_context *xo = NULL;
struct ntfs_name *name = NULL;
u64 mref;
int uname_len;
__le16 *uname;
uname_len = ntfs_nlstoucs(vol, "$ObjId", 6, &uname,
NTFS_MAX_NAME_LEN);
if (uname_len < 0)
return NULL;
/* do not use path_name_to inode - could reopen root */
dir_vi = ntfs_iget(vol->sb, FILE_Extend);
if (IS_ERR(dir_vi)) {
kmem_cache_free(ntfs_name_cache, uname);
return NULL;
}
dir_ni = NTFS_I(dir_vi);
mutex_lock_nested(&dir_ni->mrec_lock, NTFS_EXTEND_MUTEX_PARENT);
mref = ntfs_lookup_inode_by_name(dir_ni, uname, uname_len, &name);
mutex_unlock(&dir_ni->mrec_lock);
kfree(name);
kmem_cache_free(ntfs_name_cache, uname);
if (IS_ERR_MREF(mref))
goto put_dir_vi;
vi = ntfs_iget(vol->sb, MREF(mref));
if (IS_ERR(vi))
goto put_dir_vi;
xo = ntfs_index_ctx_get(NTFS_I(vi), objid_index_name, 2);
if (!xo)
iput(vi);
put_dir_vi:
iput(dir_vi);
return xo;
}
/*
* remove_object_id_index - Remove an object id index entry if attribute present
* @ni: NTFS inode structure containing the attribute
* @xo: Index context for the object id index
*
* Reads the existing object ID attribute and removes it from the index.
*
* Return: 0 on success, or a negative error code on failure.
*/
static int remove_object_id_index(struct ntfs_inode *ni, struct ntfs_index_context *xo)
{
struct object_id_index_key key = {0};
s64 size;
if (ni->data_size == 0)
return -ENODATA;
/* read the existing object id attribute */
size = ntfs_inode_attr_pread(VFS_I(ni), 0, sizeof(struct guid),
(char *)&key);
if (size != sizeof(struct guid))
return -ENODATA;
if (!ntfs_index_lookup(&key, sizeof(struct object_id_index_key), xo))
return ntfs_index_rm(xo);
return 0;
}
/*
* ntfs_delete_object_id_index - Delete an object_id index entry
* @ni: NTFS inode structure
*
* Opens the object ID index and removes the entry corresponding to the inode.
*
* Return: 0 on success, or a negative error code on failure.
*/
int ntfs_delete_object_id_index(struct ntfs_inode *ni)
{
struct ntfs_index_context *xo;
struct ntfs_inode *xoni;
struct inode *attr_vi;
int ret = 0;
attr_vi = ntfs_attr_iget(VFS_I(ni), AT_OBJECT_ID, AT_UNNAMED, 0);
if (IS_ERR(attr_vi))
return PTR_ERR(attr_vi);
/*
* read the existing object id and un-index it
*/
xo = open_object_id_index(ni->vol);
if (xo) {
xoni = xo->idx_ni;
mutex_lock_nested(&xoni->mrec_lock, NTFS_EXTEND_MUTEX_PARENT);
ret = remove_object_id_index(NTFS_I(attr_vi), xo);
if (!ret) {
ntfs_index_entry_mark_dirty(xo);
mark_mft_record_dirty(xoni);
}
ntfs_index_ctx_put(xo);
mutex_unlock(&xoni->mrec_lock);
iput(VFS_I(xoni));
}
iput(attr_vi);
return ret;
}