2 * Copyright (c) 2019 Stefan Sperling <stsp@openbsd.org>
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17 #include "got_compat.h"
20 #include <sys/queue.h>
30 #include "got_error.h"
33 #include "got_lib_lockfile.h"
35 const struct got_error
*
36 got_lockfile_lock(struct got_lockfile
**lf
, const char *path
, int dir_fd
)
38 const struct got_error
*err
= NULL
;
41 *lf
= calloc(1, sizeof(**lf
));
43 return got_error_from_errno("calloc");
46 (*lf
)->locked_path
= strdup(path
);
47 if ((*lf
)->locked_path
== NULL
) {
48 err
= got_error_from_errno("strdup");
52 if (asprintf(&(*lf
)->path
, "%s%s", path
, GOT_LOCKFILE_SUFFIX
) == -1) {
53 err
= got_error_from_errno("asprintf");
59 (*lf
)->fd
= openat(dir_fd
, (*lf
)->path
,
60 O_RDWR
| O_CREAT
| O_EXCL
| O_EXLOCK
| O_CLOEXEC
,
61 GOT_DEFAULT_FILE_MODE
);
63 (*lf
)->fd
= open((*lf
)->path
,
64 O_RDWR
| O_CREAT
| O_EXCL
| O_EXLOCK
| O_CLOEXEC
,
65 GOT_DEFAULT_FILE_MODE
);
69 if (errno
!= EEXIST
) {
70 err
= got_error_from_errno2("open", (*lf
)->path
);
74 } while (--attempts
> 0);
77 err
= got_error(GOT_ERR_LOCKFILE_TIMEOUT
);
80 got_lockfile_unlock(*lf
, dir_fd
);
86 const struct got_error
*
87 got_lockfile_unlock(struct got_lockfile
*lf
, int dir_fd
)
89 const struct got_error
*err
= NULL
;
92 if (lf
->path
&& lf
->fd
!= -1 &&
93 unlinkat(dir_fd
, lf
->path
, 0) != 0)
94 err
= got_error_from_errno("unlinkat");
95 } else if (lf
->path
&& lf
->fd
!= -1 && unlink(lf
->path
) != 0)
96 err
= got_error_from_errno("unlink");
97 if (lf
->fd
!= -1 && close(lf
->fd
) == -1 && err
== NULL
)
98 err
= got_error_from_errno("close");
100 free(lf
->locked_path
);