1 /* SPDX-License-Identifier: LGPL-2.1-or-later */
3 #include "alloc-util.h"
4 #include "chase-symlinks.h"
5 #include "dirent-util.h"
13 #include "parse-util.h"
14 #include "path-util.h"
15 #include "stat-util.h"
16 #include "string-util.h"
19 #include "xattr-util.h"
21 bool image_name_is_valid(const char *s
) {
22 if (!filename_is_valid(s
))
25 if (string_has_cc(s
, NULL
))
28 if (!utf8_is_valid(s
))
31 /* Temporary files for atomically creating new files */
32 if (startswith(s
, ".#"))
38 int path_is_extension_tree(const char *path
, const char *extension
) {
43 /* Does the path exist at all? If not, generate an error immediately. This is useful so that a missing root dir
44 * always results in -ENOENT, and we can properly distinguish the case where the whole root doesn't exist from
45 * the case where just the os-release file is missing. */
46 if (laccess(path
, F_OK
) < 0)
49 /* We use /usr/lib/extension-release.d/extension-release[.NAME] as flag for something being a system extension,
50 * and {/etc|/usr/lib}/os-release as a flag for something being an OS (when not an extension). */
51 r
= open_extension_release(path
, extension
, NULL
, NULL
);
52 if (r
== -ENOENT
) /* We got nothing */
60 int open_extension_release(const char *root
, const char *extension
, char **ret_path
, int *ret_fd
) {
61 _cleanup_free_
char *q
= NULL
;
65 const char *extension_full_path
;
67 if (!image_name_is_valid(extension
))
68 return log_debug_errno(SYNTHETIC_ERRNO(EINVAL
),
69 "The extension name %s is invalid.", extension
);
71 extension_full_path
= strjoina("/usr/lib/extension-release.d/extension-release.", extension
);
72 r
= chase_symlinks(extension_full_path
, root
, CHASE_PREFIX_ROOT
,
75 log_full_errno_zerook(LOG_DEBUG
, MIN(r
, 0), "Checking for %s: %m", extension_full_path
);
77 /* Cannot find the expected extension-release file? The image filename might have been
78 * mangled on deployment, so fallback to checking for any file in the extension-release.d
79 * directory, and return the first one with a user.extension-release xattr instead.
80 * The user.extension-release.strict xattr is checked to ensure the author of the image
81 * considers it OK if names do not match. */
83 _cleanup_free_
char *extension_release_dir_path
= NULL
;
84 _cleanup_closedir_
DIR *extension_release_dir
= NULL
;
86 r
= chase_symlinks_and_opendir("/usr/lib/extension-release.d/", root
, CHASE_PREFIX_ROOT
,
87 &extension_release_dir_path
, &extension_release_dir
);
89 return log_debug_errno(r
, "Cannot open %s/usr/lib/extension-release.d/, ignoring: %m", root
);
92 FOREACH_DIRENT(de
, extension_release_dir
, return -errno
) {
95 if (!IN_SET(de
->d_type
, DT_REG
, DT_UNKNOWN
))
98 const char *image_name
= startswith(de
->d_name
, "extension-release.");
102 if (!image_name_is_valid(image_name
)) {
103 log_debug("%s/%s is not a valid extension-release file name, ignoring.",
104 extension_release_dir_path
, de
->d_name
);
108 /* We already chased the directory, and checked that
109 * this is a real file, so we shouldn't fail to open it. */
110 _cleanup_close_
int extension_release_fd
= openat(dirfd(extension_release_dir
),
112 O_PATH
|O_CLOEXEC
|O_NOFOLLOW
);
113 if (extension_release_fd
< 0)
114 return log_debug_errno(errno
,
115 "Failed to open extension-release file %s/%s: %m",
116 extension_release_dir_path
,
119 /* Really ensure it is a regular file after we open it. */
120 if (fd_verify_regular(extension_release_fd
) < 0) {
121 log_debug("%s/%s is not a regular file, ignoring.", extension_release_dir_path
, de
->d_name
);
125 /* No xattr or cannot parse it? Then skip this. */
126 _cleanup_free_
char *extension_release_xattr
= NULL
;
127 k
= fgetxattr_malloc(extension_release_fd
, "user.extension-release.strict", &extension_release_xattr
);
128 if (k
< 0 && !ERRNO_IS_NOT_SUPPORTED(k
) && k
!= -ENODATA
)
130 "%s/%s: Failed to read 'user.extension-release.strict' extended attribute from file: %m",
131 extension_release_dir_path
, de
->d_name
);
133 log_debug("%s/%s does not have user.extension-release.strict xattr, ignoring.", extension_release_dir_path
, de
->d_name
);
137 /* Explicitly set to request strict matching? Skip it. */
138 k
= parse_boolean(extension_release_xattr
);
141 "%s/%s: Failed to parse 'user.extension-release.strict' extended attribute from file: %m",
142 extension_release_dir_path
, de
->d_name
);
144 log_debug("%s/%s: 'user.extension-release.strict' attribute is true, ignoring file.",
145 extension_release_dir_path
, de
->d_name
);
149 log_debug("%s/%s: 'user.extension-release.strict' attribute is falseā¦",
150 extension_release_dir_path
, de
->d_name
);
152 /* We already found what we were looking for, but there's another candidate?
153 * We treat this as an error, as we want to enforce that there are no ambiguities
154 * in case we are in the fallback path.*/
160 r
= 0; /* Found it! */
163 fd
= TAKE_FD(extension_release_fd
);
166 q
= path_join(extension_release_dir_path
, de
->d_name
);
175 FOREACH_STRING(p
, "/etc/os-release", "/usr/lib/os-release") {
176 r
= chase_symlinks(p
, root
, CHASE_PREFIX_ROOT
,
177 ret_path
? &q
: NULL
,
178 ret_fd
? &fd
: NULL
);
189 /* Convert the O_PATH fd into a proper, readable one */
190 real_fd
= fd_reopen(fd
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
);
199 *ret_path
= TAKE_PTR(q
);
204 int fopen_extension_release(const char *root
, const char *extension
, char **ret_path
, FILE **ret_file
) {
205 _cleanup_free_
char *p
= NULL
;
206 _cleanup_close_
int fd
= -1;
211 return open_extension_release(root
, extension
, ret_path
, NULL
);
213 r
= open_extension_release(root
, extension
, ret_path
? &p
: NULL
, &fd
);
217 f
= take_fdopen(&fd
, "r");
222 *ret_path
= TAKE_PTR(p
);
228 static int parse_release_internal(const char *root
, const char *extension
, va_list ap
) {
229 _cleanup_fclose_
FILE *f
= NULL
;
230 _cleanup_free_
char *p
= NULL
;
233 r
= fopen_extension_release(root
, extension
, &p
, &f
);
237 return parse_env_filev(f
, p
, ap
);
240 int _parse_extension_release(const char *root
, const char *extension
, ...) {
244 va_start(ap
, extension
);
245 r
= parse_release_internal(root
, extension
, ap
);
251 int _parse_os_release(const char *root
, ...) {
256 r
= parse_release_internal(root
, NULL
, ap
);
262 int load_os_release_pairs(const char *root
, char ***ret
) {
263 _cleanup_fclose_
FILE *f
= NULL
;
264 _cleanup_free_
char *p
= NULL
;
267 r
= fopen_os_release(root
, &p
, &f
);
271 return load_env_file_pairs(f
, p
, ret
);
274 int load_os_release_pairs_with_prefix(const char *root
, const char *prefix
, char ***ret
) {
275 _cleanup_strv_free_
char **os_release_pairs
= NULL
, **os_release_pairs_prefixed
= NULL
;
279 r
= load_os_release_pairs(root
, &os_release_pairs
);
283 STRV_FOREACH_PAIR(p
, q
, os_release_pairs
) {
286 /* We strictly return only the four main ID fields and ignore the rest */
287 if (!STR_IN_SET(*p
, "ID", "VERSION_ID", "BUILD_ID", "VARIANT_ID"))
291 line
= strjoin(prefix
, *p
, "=", *q
);
294 r
= strv_consume(&os_release_pairs_prefixed
, line
);
299 *ret
= TAKE_PTR(os_release_pairs_prefixed
);
304 int load_extension_release_pairs(const char *root
, const char *extension
, char ***ret
) {
305 _cleanup_fclose_
FILE *f
= NULL
;
306 _cleanup_free_
char *p
= NULL
;
309 r
= fopen_extension_release(root
, extension
, &p
, &f
);
313 return load_env_file_pairs(f
, p
, ret
);