74 lines
2.9 KiB
74 lines
2.9 KiB
lockfile API |
|
============ |
|
|
|
The lockfile API serves two purposes: |
|
|
|
* Mutual exclusion. When we write out a new index file, first |
|
we create a new file `$GIT_DIR/index.lock`, write the new |
|
contents into it, and rename it to the final destination |
|
`$GIT_DIR/index`. We try to create the `$GIT_DIR/index.lock` |
|
file with O_EXCL so that we can notice and fail when somebody |
|
else is already trying to update the index file. |
|
|
|
* Automatic cruft removal. After we create the "lock" file, we |
|
may decide to `die()`, and we would want to make sure that we |
|
remove the file that has not been committed to its final |
|
destination. This is done by remembering the lockfiles we |
|
created in a linked list and cleaning them up from an |
|
`atexit(3)` handler. Outstanding lockfiles are also removed |
|
when the program dies on a signal. |
|
|
|
|
|
The functions |
|
------------- |
|
|
|
hold_lock_file_for_update:: |
|
|
|
Take a pointer to `struct lock_file`, the filename of |
|
the final destination (e.g. `$GIT_DIR/index`) and a flag |
|
`die_on_error`. Attempt to create a lockfile for the |
|
destination and return the file descriptor for writing |
|
to the file. If `die_on_error` flag is true, it dies if |
|
a lock is already taken for the file; otherwise it |
|
returns a negative integer to the caller on failure. |
|
|
|
commit_lock_file:: |
|
|
|
Take a pointer to the `struct lock_file` initialized |
|
with an earlier call to `hold_lock_file_for_update()`, |
|
close the file descriptor and rename the lockfile to its |
|
final destination. Returns 0 upon success, a negative |
|
value on failure to close(2) or rename(2). |
|
|
|
rollback_lock_file:: |
|
|
|
Take a pointer to the `struct lock_file` initialized |
|
with an earlier call to `hold_lock_file_for_update()`, |
|
close the file descriptor and remove the lockfile. |
|
|
|
close_lock_file:: |
|
Take a pointer to the `struct lock_file` initialized |
|
with an earlier call to `hold_lock_file_for_update()`, |
|
and close the file descriptor. Returns 0 upon success, |
|
a negative value on failure to close(2). |
|
|
|
Because the structure is used in an `atexit(3)` handler, its |
|
storage has to stay throughout the life of the program. It |
|
cannot be an auto variable allocated on the stack. |
|
|
|
Call `commit_lock_file()` or `rollback_lock_file()` when you are |
|
done writing to the file descriptor. If you do not call either |
|
and simply `exit(3)` from the program, an `atexit(3)` handler |
|
will close and remove the lockfile. |
|
|
|
If you need to close the file descriptor you obtained from |
|
`hold_lock_file_for_update` function yourself, do so by calling |
|
`close_lock_file()`. You should never call `close(2)` yourself! |
|
Otherwise the `struct |
|
lock_file` structure still remembers that the file descriptor |
|
needs to be closed, and a later call to `commit_lock_file()` or |
|
`rollback_lock_file()` will result in duplicate calls to |
|
`close(2)`. Worse yet, if you `close(2)`, open another file |
|
descriptor for completely different purpose, and then call |
|
`commit_lock_file()` or `rollback_lock_file()`, they may close |
|
that unrelated file descriptor.
|
|
|