diff options
-rw-r--r-- | README.bg.md | 9 | ||||
-rw-r--r-- | README.de.md | 5 | ||||
-rw-r--r-- | README.el.md | 5 | ||||
-rw-r--r-- | README.fr.md | 5 | ||||
-rw-r--r-- | README.it.md | 5 | ||||
-rw-r--r-- | README.ja.md | 3 | ||||
-rw-r--r-- | README.md | 3 | ||||
-rw-r--r-- | README.pt.md | 5 | ||||
-rw-r--r-- | README.ru.md | 9 | ||||
-rw-r--r-- | README.uk.md | 5 | ||||
-rw-r--r-- | completions/bash/swaylock | 66 | ||||
-rw-r--r-- | completions/fish/swaylock.fish | 39 | ||||
-rw-r--r-- | completions/zsh/_swaylock | 40 | ||||
-rw-r--r-- | meson.build | 5 | ||||
-rw-r--r-- | swaylock/main.c | 1057 | ||||
-rw-r--r-- | swaylock/meson.build | 55 | ||||
-rw-r--r-- | swaylock/pam.c | 62 | ||||
-rw-r--r-- | swaylock/pam/swaylock.freebsd | 6 | ||||
-rw-r--r-- | swaylock/pam/swaylock.linux | 6 | ||||
-rw-r--r-- | swaylock/password.c | 190 | ||||
-rw-r--r-- | swaylock/render.c | 189 | ||||
-rw-r--r-- | swaylock/seat.c | 178 | ||||
-rw-r--r-- | swaylock/shadow.c | 155 | ||||
-rw-r--r-- | swaylock/swaylock.1.scd | 167 |
24 files changed, 14 insertions, 2255 deletions
diff --git a/README.bg.md b/README.bg.md index 87a15bed..dcdefc7e 100644 --- a/README.bg.md +++ b/README.bg.md @@ -45,16 +45,13 @@ Sway съществува в репотата (хранилищата) на мн * pango * cairo * gdk-pixbuf2 * -* pam ** -* dbus >= 1.10 *** +* dbus >= 1.10 ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (Нужен за man страници) * git -_\*Нужен само за swaybar, swaybg и swaylock_ +_\*Нужен само за swaybar, swaybg_ -_\*\*Нужен само за swaylock_ - -_\*\*\*Нужен само за tray_ +_\*\*Нужен само за tray_ Изпълнете следните команди: diff --git a/README.de.md b/README.de.md index bf92b196..e8198013 100644 --- a/README.de.md +++ b/README.de.md @@ -56,13 +56,10 @@ Abhängigkeiten: * pango * cairo * gdk-pixbuf2 * -* pam ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (erforderlich für man pages) * git -_\*Nur erforderlich für swaybar, swaybg, und swaylock_ - -_\*\*Nur erforderlich für swaylock_ +_\*Nur erforderlich für swaybar, swaybg_ Führe diese Befehle aus: diff --git a/README.el.md b/README.el.md index 79acd422..bc2360ca 100644 --- a/README.el.md +++ b/README.el.md @@ -48,13 +48,10 @@ To username μου στο Freenode είναι kon14 και θα με βρείτ * pango * cairo * gdk-pixbuf2 * -* pam ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (required for man pages) * git -_\*Απαιτείται μόνο για swaybar, swaybg, and swaylock_ - -_\*\*Απαιτείται μόνο για swaylock_ +_\*Απαιτείται μόνο για swaybar, swaybg_ Εκτελέστε αυτές τις εντολές: diff --git a/README.fr.md b/README.fr.md index 8ad4d3b6..ad627ba5 100644 --- a/README.fr.md +++ b/README.fr.md @@ -50,13 +50,10 @@ Installez les dépendances : * pango * cairo * gdk-pixbuf2 * -* pam ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (requis pour les pages man) * git -_\*Uniquement requis pour swaybar, swaybg, and swaylock_ - -_\*\*Uniquement requis pour swaylock_ +_\*Uniquement requis pour swaybar, swaybg_ Exécutez ces commandes : diff --git a/README.it.md b/README.it.md index 8a83bc78..bfe09920 100644 --- a/README.it.md +++ b/README.it.md @@ -50,13 +50,10 @@ Installa queste dipendenze: * pango * cairo * gdk-pixbuf2 * -* pam ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (rrichiesto per man pages) * git -_\*Richiesto solo per swaybar, swaybg, e swaylock_ - -_\*\*Richiesto solo per swaylock_ +_\*Richiesto solo per swaybar, swaybg_ Esegui questi comandi: diff --git a/README.ja.md b/README.ja.md index 42237136..800daa62 100644 --- a/README.ja.md +++ b/README.ja.md @@ -37,13 +37,12 @@ Swayは沢山のディストリビューションで提供されています。" * pango * cairo * gdk-pixbuf2 \*\* -* pam (オプション: swaylockとPAMで必要) * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (manで必要です) \* * git \* _\*コンパイルの時_ -_\*\*オプション: swaybgとswaylockでのみ必要です_ +_\*\*オプション: swaybgでのみ必要です_ 次のコマンドを実行してください: @@ -40,13 +40,12 @@ Install dependencies: * pango * cairo * gdk-pixbuf2 \*\* -* pam (optional: PAM support for swaylock) * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (optional: man pages) \* * git \* _\*Compile-time dep_ -_\*\*optional: required for swaybg and swaylock_ +_\*\*optional: required for swaybg_ Run these commands: diff --git a/README.pt.md b/README.pt.md index 8cdfa548..0f2a3cd4 100644 --- a/README.pt.md +++ b/README.pt.md @@ -57,13 +57,10 @@ Antes de iniciar a compilação, instale as dependências: * pango * cairo * gdk-pixbuf2 * -* pam ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (man pages) * git -_\*Dependência apenas de swaybar, swaybg, e swaylock_ - -_\*\*Dependência apenas de swaylock_ +_\*Dependência apenas de swaybar, swaybg_ Para compilar, execute estes comandos: diff --git a/README.ru.md b/README.ru.md index 255e36aa..c66976e5 100644 --- a/README.ru.md +++ b/README.ru.md @@ -50,16 +50,13 @@ Sway доступен во многих дистрибутивах и наход * pango * cairo * gdk-pixbuf2 * -* pam ** -* dbus >= 1.10 *** +* dbus >= 1.10 ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (required for man pages) * git -_\*Требуется только для swaybar, swaybg и swaylock_ +_\*Требуется только для swaybar, swaybg_ -_\*\*Требуется только для swaylock_ - -_\*\*\*Требуется только для tray_ +_\*\*Требуется только для tray_ Выполните следующие команды: diff --git a/README.uk.md b/README.uk.md index 5e9345b3..4c378a89 100644 --- a/README.uk.md +++ b/README.uk.md @@ -57,13 +57,10 @@ Sway доступний у багатьох дистрибутивах Linux (а * pango * cairo * gdk-pixbuf2 * -* pam ** * [scdoc](https://git.sr.ht/~sircmpwn/scdoc) (required for man pages) * git -_\*Лише для swaybar, swaybg та swaylock_ - -_\*\*Лише для swaylock_ +_\*Лише для swaybar, swaybg_ Виконайте ці команди: diff --git a/completions/bash/swaylock b/completions/bash/swaylock deleted file mode 100644 index 33925480..00000000 --- a/completions/bash/swaylock +++ /dev/null @@ -1,66 +0,0 @@ -# swaylock(1) completion - -_swaylock() -{ - local cur prev - _get_comp_words_by_ref -n : cur prev - - short=( - -h - -c - -s - -t - -v - -i - -u - -f - ) - - long=( - --help - --color - --scaling - --tiling - --version - --image - --no-unlock-indicator - --daemonize - ) - - scaling=( - 'stretch' - 'fill' - 'fit' - 'center' - 'tile' - ) - - case $prev in - -c|--color) - return - ;; - --scaling) - COMPREPLY=($(compgen -W "${scaling[*]}" -- "$cur")) - return - ;; - -i|--image) - if grep -q : <<< "$cur"; then - output="${cur%%:*}:" - cur="${cur#*:}" - else - output= - fi - COMPREPLY=($(compgen -f -- "$cur")) - return - ;; - esac - - if [[ $cur == --* ]]; then - COMPREPLY=($(compgen -W "${long[*]}" -- "$cur")) - else - COMPREPLY=($(compgen -W "${short[*]}" -- "$cur")) - COMPREPLY+=($(compgen -W "${long[*]}" -- "$cur")) - fi - -} && -complete -F _swaylock swaylock diff --git a/completions/fish/swaylock.fish b/completions/fish/swaylock.fish deleted file mode 100644 index 99dff48f..00000000 --- a/completions/fish/swaylock.fish +++ /dev/null @@ -1,39 +0,0 @@ -# swaylock(1) completion - -complete -c swaylock -s C -l config --description 'The config file to use. Default: $HOME/.swaylock/config, $XDG_CONFIG_HOME/swaylock/config, and SYSCONFDIR/swaylock/config.' -complete -c swaylock -s h -l help --description "Show help message and quit." -complete -c swaylock -s f -l daemonize --description "Fork into the background after spawning. Note: this is the default bahavior of i3lock." -complete -c swaylock -s v -l version --description "Show the version number and quit." -complete -c swaylock -s s -l socket --description "Use the specified socket path. Otherwise, swaymsg will as sway where the socket is (which is the value of $SWAYSOCK, then of $I350CK)." -complete -c swaylock -s e -l ignore-empty-password --description 'When an empty password is provided by the user, do not validate it.' - -# Appearance -complete -c swaylock -s u -l no-unlock-indicator --description "Disable the unlock indicator." -complete -c swaylock -s i -l image --description "Display the given image, optionally on the given output. Use -c to set a background color." -complete -c swaylock -s s -l scaling --description "Scaling mode for images: stretch, fill, fit, center, or tile." -complete -c swaylock -s t -l tiling --description "Same as --scaling=tile." -complete -c swaylock -s c -l color --description "Turn the screen into the given color. If -i is used, this sets the background of the image into the given color. Defaults to white (ffffff), or transparent (00000000) if an image is in use." -complete -c swaylock -l bs-hl-color --description 'Sets the color of backspace highlight segments.' -complete -c swaylock -l font --description 'Sets the font of the text inside the indicator.' -complete -c swaylock -l indicator-radius --description 'Sets the radius of the indicator to radius pixels. Default: 50' -complete -c swaylock -l indicator-thickness --description 'Sets the thickness of the indicator to thickness pixels. Default: 10' -complete -c swaylock -l inside-color --description 'Sets the color of the inside of the indicator when typing or idle.' -complete -c swaylock -l inside-clear-color --description 'Sets the color of the inside of the indicator when cleared.' -complete -c swaylock -l inside-ver-color --description 'Sets the color of the inside of the indicator when verifying.' -complete -c swaylock -l inside-wrong-color --description 'Sets the color of the inside of the indicator when invalid.' -complete -c swaylock -l key-hl-color --description 'Sets the color of key press highlight segments.' -complete -c swaylock -l line-color --description 'Sets the color of the lines that separate the inside and outside of the indicator when typing or idle.' -complete -c swaylock -l line-clear-color --description 'Sets the color of the lines that separate the inside and outside of the indicator when cleared.' -complete -c swaylock -l line-ver-color --description 'Sets the color of the lines that separate the inside and outside of the indicator when verifying.' -complete -c swaylock -l line-wrong-color --description 'Sets the color of the lines that separate the inside and outside of the indicator when invalid.' -complete -c swaylock -s n -l line-uses-inside --description 'Use the color of the inside of the indicator for the line separating the inside and outside of the indicator.' -complete -c swaylock -s r -l line-uses-ring --description 'Use the outer ring\'s color for the line separating the inside and outside of the indicator.' -complete -c swaylock -l ring-color --description 'Sets the color of the outside of the indicator when typing or idle.' -complete -c swaylock -l ring-clear-color --description 'Sets the color of the outside of the indicator when cleared.' -complete -c swaylock -l ring-ver-color --description 'Sets the color of the outside of the indicator when verifying.' -complete -c swaylock -l ring-wrong-color --description 'Sets the color of the outside of the indicator when invalid.' -complete -c swaylock -l separator-color --description 'Sets the color of the lines that separate highlight segments.' -complete -c swaylock -l text-color --description 'Sets the color of the text inside the indicator when typing or idle.' -complete -c swaylock -l text-clear-color --description 'Sets the color of the text inside the indicator when cleared.' -complete -c swaylock -l text-ver-color --description 'Sets the color of the text inside the indicator when verifying.' -complete -c swaylock -l text-wrong-color --description 'Sets the color of the text inside the indicator when invalid.' diff --git a/completions/zsh/_swaylock b/completions/zsh/_swaylock deleted file mode 100644 index 9bc84ec9..00000000 --- a/completions/zsh/_swaylock +++ /dev/null @@ -1,40 +0,0 @@ -#compdef swaylock -# -# Completion script for swaylock -# - -_arguments -s \ - '(-C --config)'{-C,--config}'[Path to the config file]:filename:_files' \ - '(-c --color)'{-c,--color}'[Turn the screen into the given color instead of white]:color:' \ - '(-e --ignore-empty-password)'{-e,--ignore-empty-password}'[When an empty password is provided, do not validate it]' \ - '(-f --daemonize)'{-f,--daemonize}'[Detach from the controlling terminal after locking]' \ - '(-h --help)'{-h,--help}'[Show help message and quit]' \ - '(-i --image)'{-i,--image}'[Display an image]:filename:_files' \ - '(-s --scaling)'{-s,--scaling}'[Scaling mode]:mode:(stretch fill fit center tile)' \ - '(-t --tiling)'{-t,--tiling}'[Same as --scaling=tile]' \ - '(-u --no-unlock-indicator)'{-u,--no-unlock-indicator}'[Disable the unlock indicator]' \ - '(-v --version)'{-v,--version}'[Show the version number and quit]' \ - '(--bs-hl-color)'--bs-hl-color'[Sets the color of backspace highlights segments]:color:' \ - '(--font)'--font'[Sets the font of the text]:font:' \ - '(--indicator-radius)'--indicator-radius'[Sets the indicator radius]:radius:' \ - '(--indicator-thickness)'--indicator-thickness'[Sets the indicator thickness]:thickness:' \ - '(--inside-color)'--inside-color'[Sets the color of the inside of the indicator]:color:' \ - '(--inside-clear-color)'--inside-clear-color'[Sets the color of the inside of the indicator when cleared]:color:' \ - '(--inside-clear-color)'--inside-clear-color'[Sets the color of the inside of the indicator when verifying]:color:' \ - '(--inside-wrong-color)'--inside-wrong-color'[Sets the color of the inside of the indicator when invalid]:color:' \ - '(--key-hl-color)'--key-hl-color'[Sets the color of the key press highlight segments]:color:' \ - '(--line-color)'--line-color'[Sets the color of the line between the inside and ring]:color:' \ - '(--line-clear-color)'--line-clear-color'[Sets the color of the line between the inside and ring when cleared]:color:' \ - '(--line-ver-color)'--line-ver-color'[Sets the color of the line between the inside and ring when verifying]:color:' \ - '(--line-wrong-color)'--line-wrong-color'[Sets the color of the line between the inside and ring when invalid]:color:' \ - '(-n --line-uses-inside)'{-n,--line-uses-inside}'[Use the inside color for the line between the inside and ring]' \ - '(-r --line-uses-ring)'{-r,--line--uses-ring}'[Use the ring color for the line between the inside and ring]' \ - '(--ring-color)'--ring-color'[Sets the color of the ring of the indicator]:color:' \ - '(--ring-clear-color)'--ring-clear-color'[Sets the color of the ring of the indicator when cleared]:color:' \ - '(--ring-ver-color)'--ring-ver-color'[Sets the color of the ring of the indicator when verifying]:color:' \ - '(--ring-wrong-color)'--ring-wrong-color'[Sets the color of the ring of the indicator when invalid]:color:' \ - '(--separator-color)'--separator-color'[Sets the color of the lines that separate highlight segments]:color:' \ - '(--text-color)'--text-color'[Sets the color of the text]:color:' \ - '(--text-clear-color)'--text-clear-color'[Sets the color of the text when cleared]:color:' \ - '(--text-ver-color)'--text-ver-color'[Sets the color of the text when verifying]:color:' \ - '(--text-wrong-color)'--text-wrong-color'[Sets the color of the text when invalid]:color:'
\ No newline at end of file diff --git a/meson.build b/meson.build index bffbe312..520a87ad 100644 --- a/meson.build +++ b/meson.build @@ -84,7 +84,6 @@ if scdoc.found() 'sway/sway-bar.5.scd', 'sway/sway-input.5.scd', 'sway/sway-output.5.scd', - 'swaylock/swaylock.1.scd', 'swaymsg/swaymsg.1.scd', 'swaynag/swaynag.1.scd', 'swaynag/swaynag.5.scd', @@ -146,7 +145,6 @@ subdir('client') subdir('swaybg') subdir('swaybar') subdir('swaynag') -subdir('swaylock') config = configuration_data() config.set('datadir', join_paths(prefix, datadir)) @@ -200,7 +198,6 @@ endif if (get_option('zsh-completions')) zsh_files = files( 'completions/zsh/_sway', - 'completions/zsh/_swaylock', 'completions/zsh/_swaymsg', ) zsh_install_dir = datadir + '/zsh/site-functions' @@ -212,7 +209,6 @@ if (get_option('bash-completions')) bash_files = files( 'completions/bash/sway', 'completions/bash/swaybar', - 'completions/bash/swaylock', 'completions/bash/swaymsg', ) bash_install_dir = datadir + '/bash-completion/completions' @@ -223,7 +219,6 @@ endif if (get_option('fish-completions')) fish_files = files( 'completions/fish/sway.fish', - 'completions/fish/swaylock.fish', 'completions/fish/swaymsg.fish', 'completions/fish/swaynag.fish', ) diff --git a/swaylock/main.c b/swaylock/main.c deleted file mode 100644 index 0b167da1..00000000 --- a/swaylock/main.c +++ /dev/null @@ -1,1057 +0,0 @@ -#define _POSIX_C_SOURCE 200809L -#include <assert.h> -#include <ctype.h> -#include <errno.h> -#include <fcntl.h> -#include <getopt.h> -#include <poll.h> -#include <stdbool.h> -#include <stdio.h> -#include <stdlib.h> -#include <string.h> -#include <sys/mman.h> -#include <sys/stat.h> -#include <time.h> -#include <unistd.h> -#include <wayland-client.h> -#include <wordexp.h> -#include <wlr/util/log.h> -#include "swaylock/seat.h" -#include "swaylock/swaylock.h" -#include "background-image.h" -#include "pool-buffer.h" -#include "cairo.h" -#include "log.h" -#include "loop.h" -#include "stringop.h" -#include "util.h" -#include "wlr-input-inhibitor-unstable-v1-client-protocol.h" -#include "wlr-layer-shell-unstable-v1-client-protocol.h" -#include "xdg-output-unstable-v1-client-protocol.h" - -void sway_terminate(int exit_code) { - exit(exit_code); -} - -static void daemonize(void) { - int fds[2]; - if (pipe(fds) != 0) { - wlr_log(WLR_ERROR, "Failed to pipe"); - exit(1); - } - if (fork() == 0) { - setsid(); - close(fds[0]); - int devnull = open("/dev/null", O_RDWR); - dup2(STDOUT_FILENO, devnull); - dup2(STDERR_FILENO, devnull); - close(devnull); - uint8_t success = 0; - if (chdir("/") != 0) { - write(fds[1], &success, 1); - exit(1); - } - success = 1; - if (write(fds[1], &success, 1) != 1) { - exit(1); - } - close(fds[1]); - } else { - close(fds[1]); - uint8_t success; - if (read(fds[0], &success, 1) != 1 || !success) { - wlr_log(WLR_ERROR, "Failed to daemonize"); - exit(1); - } - close(fds[0]); - exit(0); - } -} - -static void destroy_surface(struct swaylock_surface *surface) { - wl_list_remove(&surface->link); - if (surface->layer_surface != NULL) { - zwlr_layer_surface_v1_destroy(surface->layer_surface); - } - if (surface->surface != NULL) { - wl_surface_destroy(surface->surface); - } - destroy_buffer(&surface->buffers[0]); - destroy_buffer(&surface->buffers[1]); - wl_output_destroy(surface->output); - free(surface); -} - -static const struct zwlr_layer_surface_v1_listener layer_surface_listener; - -static cairo_surface_t *select_image(struct swaylock_state *state, - struct swaylock_surface *surface); - -static bool surface_is_opaque(struct swaylock_surface *surface) { - if (surface->image) { - return cairo_surface_get_content(surface->image) == CAIRO_CONTENT_COLOR; - } - return (surface->state->args.colors.background & 0xff) == 0xff; -} - -static void create_layer_surface(struct swaylock_surface *surface) { - struct swaylock_state *state = surface->state; - - surface->image = select_image(state, surface); - - surface->surface = wl_compositor_create_surface(state->compositor); - assert(surface->surface); - - surface->layer_surface = zwlr_layer_shell_v1_get_layer_surface( - state->layer_shell, surface->surface, surface->output, - ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY, "lockscreen"); - assert(surface->layer_surface); - - zwlr_layer_surface_v1_set_size(surface->layer_surface, 0, 0); - zwlr_layer_surface_v1_set_anchor(surface->layer_surface, - ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP | - ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT | - ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM | - ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT); - zwlr_layer_surface_v1_set_exclusive_zone(surface->layer_surface, -1); - zwlr_layer_surface_v1_set_keyboard_interactivity( - surface->layer_surface, true); - zwlr_layer_surface_v1_add_listener(surface->layer_surface, - &layer_surface_listener, surface); - - if (surface_is_opaque(surface) && - surface->state->args.mode != BACKGROUND_MODE_CENTER && - surface->state->args.mode != BACKGROUND_MODE_FIT) { - struct wl_region *region = - wl_compositor_create_region(surface->state->compositor); - wl_region_add(region, 0, 0, INT32_MAX, INT32_MAX); - wl_surface_set_opaque_region(surface->surface, region); - wl_region_destroy(region); - } - - wl_surface_commit(surface->surface); -} - -static void layer_surface_configure(void *data, - struct zwlr_layer_surface_v1 *layer_surface, - uint32_t serial, uint32_t width, uint32_t height) { - struct swaylock_surface *surface = data; - surface->width = width; - surface->height = height; - zwlr_layer_surface_v1_ack_configure(layer_surface, serial); - render_frame(surface); -} - -static void layer_surface_closed(void *data, - struct zwlr_layer_surface_v1 *layer_surface) { - struct swaylock_surface *surface = data; - destroy_surface(surface); -} - -static const struct zwlr_layer_surface_v1_listener layer_surface_listener = { - .configure = layer_surface_configure, - .closed = layer_surface_closed, -}; - -static const struct wl_callback_listener surface_frame_listener; - -static void surface_frame_handle_done(void *data, struct wl_callback *callback, - uint32_t time) { - struct swaylock_surface *surface = data; - - wl_callback_destroy(callback); - surface->frame_pending = false; - - if (surface->dirty) { - // Schedule a frame in case the surface is damaged again - struct wl_callback *callback = wl_surface_frame(surface->surface); - wl_callback_add_listener(callback, &surface_frame_listener, surface); - surface->frame_pending = true; - - render_frame(surface); - surface->dirty = false; - } -} - -static const struct wl_callback_listener surface_frame_listener = { - .done = surface_frame_handle_done, -}; - -void damage_surface(struct swaylock_surface *surface) { - surface->dirty = true; - if (surface->frame_pending) { - return; - } - - struct wl_callback *callback = wl_surface_frame(surface->surface); - wl_callback_add_listener(callback, &surface_frame_listener, surface); - surface->frame_pending = true; - wl_surface_commit(surface->surface); -} - -void damage_state(struct swaylock_state *state) { - struct swaylock_surface *surface; - wl_list_for_each(surface, &state->surfaces, link) { - damage_surface(surface); - } -} - -static void handle_wl_output_geometry(void *data, struct wl_output *wl_output, - int32_t x, int32_t y, int32_t width_mm, int32_t height_mm, - int32_t subpixel, const char *make, const char *model, - int32_t transform) { - struct swaylock_surface *surface = data; - surface->subpixel = subpixel; - if (surface->state->run_display) { - damage_surface(surface); - } -} - -static void handle_wl_output_mode(void *data, struct wl_output *output, - uint32_t flags, int32_t width, int32_t height, int32_t refresh) { - // Who cares -} - -static void handle_wl_output_done(void *data, struct wl_output *output) { - // Who cares -} - -static void handle_wl_output_scale(void *data, struct wl_output *output, - int32_t factor) { - struct swaylock_surface *surface = data; - surface->scale = factor; - if (surface->state->run_display) { - damage_surface(surface); - } -} - -struct wl_output_listener _wl_output_listener = { - .geometry = handle_wl_output_geometry, - .mode = handle_wl_output_mode, - .done = handle_wl_output_done, - .scale = handle_wl_output_scale, -}; - -static void handle_xdg_output_logical_size(void *data, struct zxdg_output_v1 *output, - int width, int height) { - // Who cares -} - -static void handle_xdg_output_logical_position(void *data, - struct zxdg_output_v1 *output, int x, int y) { - // Who cares -} - -static void handle_xdg_output_name(void *data, struct zxdg_output_v1 *output, - const char *name) { - wlr_log(WLR_DEBUG, "output name is %s", name); - struct swaylock_surface *surface = data; - surface->xdg_output = output; - surface->output_name = strdup(name); -} - -static void handle_xdg_output_description(void *data, struct zxdg_output_v1 *output, - const char *description) { - // Who cares -} - -static void handle_xdg_output_done(void *data, struct zxdg_output_v1 *output) { - // Who cares -} - -struct zxdg_output_v1_listener _xdg_output_listener = { - .logical_position = handle_xdg_output_logical_position, - .logical_size = handle_xdg_output_logical_size, - .done = handle_xdg_output_done, - .name = handle_xdg_output_name, - .description = handle_xdg_output_description, -}; - -static void handle_global(void *data, struct wl_registry *registry, - uint32_t name, const char *interface, uint32_t version) { - struct swaylock_state *state = data; - if (strcmp(interface, wl_compositor_interface.name) == 0) { - state->compositor = wl_registry_bind(registry, name, - &wl_compositor_interface, 3); - } else if (strcmp(interface, wl_shm_interface.name) == 0) { - state->shm = wl_registry_bind(registry, name, - &wl_shm_interface, 1); - } else if (strcmp(interface, wl_seat_interface.name) == 0) { - struct wl_seat *seat = wl_registry_bind( - registry, name, &wl_seat_interface, 3); - struct swaylock_seat *swaylock_seat = - calloc(1, sizeof(struct swaylock_seat)); - swaylock_seat->state = state; - wl_seat_add_listener(seat, &seat_listener, swaylock_seat); - } else if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) { - state->layer_shell = wl_registry_bind( - registry, name, &zwlr_layer_shell_v1_interface, 1); - } else if (strcmp(interface, zwlr_input_inhibit_manager_v1_interface.name) == 0) { - state->input_inhibit_manager = wl_registry_bind( - registry, name, &zwlr_input_inhibit_manager_v1_interface, 1); - } else if (strcmp(interface, zxdg_output_manager_v1_interface.name) == 0) { - state->zxdg_output_manager = wl_registry_bind( - registry, name, &zxdg_output_manager_v1_interface, 2); - } else if (strcmp(interface, wl_output_interface.name) == 0) { - struct swaylock_surface *surface = - calloc(1, sizeof(struct swaylock_surface)); - surface->state = state; - surface->output = wl_registry_bind(registry, name, - &wl_output_interface, 3); - surface->output_global_name = name; - wl_output_add_listener(surface->output, &_wl_output_listener, surface); - wl_list_insert(&state->surfaces, &surface->link); - - if (state->run_display) { - create_layer_surface(surface); - wl_display_roundtrip(state->display); - } - } -} - -static void handle_global_remove(void *data, struct wl_registry *registry, - uint32_t name) { - struct swaylock_state *state = data; - struct swaylock_surface *surface; - wl_list_for_each(surface, &state->surfaces, link) { - if (surface->output_global_name == name) { - destroy_surface(surface); - break; - } - } -} - -static const struct wl_registry_listener registry_listener = { - .global = handle_global, - .global_remove = handle_global_remove, -}; - -static cairo_surface_t *select_image(struct swaylock_state *state, - struct swaylock_surface *surface) { - struct swaylock_image *image; - cairo_surface_t *default_image = NULL; - wl_list_for_each(image, &state->images, link) { - if (lenient_strcmp(image->output_name, surface->output_name) == 0) { - return image->cairo_surface; - } else if (!image->output_name) { - default_image = image->cairo_surface; - } - } - return default_image; -} - -static void load_image(char *arg, struct swaylock_state *state) { - // [<output>:]<path> - struct swaylock_image *image = calloc(1, sizeof(struct swaylock_image)); - char *separator = strchr(arg, ':'); - if (separator) { - *separator = '\0'; - image->output_name = strdup(arg); - image->path = strdup(separator + 1); - } else { - image->output_name = NULL; - image->path = strdup(arg); - } - - struct swaylock_image *iter_image, *temp; - wl_list_for_each_safe(iter_image, temp, &state->images, link) { - if (lenient_strcmp(iter_image->output_name, image->output_name) == 0) { - if (image->output_name) { - wlr_log(WLR_DEBUG, - "Replacing image defined for output %s with %s", - image->output_name, image->path); - } else { - wlr_log(WLR_DEBUG, "Replacing default image with %s", - image->path); - } - wl_list_remove(&iter_image->link); - free(iter_image->cairo_surface); - free(iter_image->output_name); - free(iter_image->path); - free(iter_image); - break; - } - } - - // Bash doesn't replace the ~ with $HOME if the output name is supplied - wordexp_t p; - if (wordexp(image->path, &p, 0) == 0) { - free(image->path); - image->path = strdup(p.we_wordv[0]); - wordfree(&p); - } - - // Load the actual image - image->cairo_surface = load_background_image(image->path); - if (!image->cairo_surface) { - free(image); - return; - } - wl_list_insert(&state->images, &image->link); - wlr_log(WLR_DEBUG, "Loaded image %s for output %s", - image->path, image->output_name ? image->output_name : "*"); -} - -static void set_default_colors(struct swaylock_colors *colors) { - colors->background = 0xFFFFFFFF; - colors->bs_highlight = 0xDB3300FF; - colors->key_highlight = 0x33DB00FF; - colors->caps_lock_bs_highlight = 0xDB3300FF; - colors->caps_lock_key_highlight = 0x33DB00FF; - colors->separator = 0x000000FF; - colors->inside = (struct swaylock_colorset){ - .input = 0x000000C0, - .cleared = 0xE5A445C0, - .caps_lock = 0x000000C0, - .verifying = 0x0072FFC0, - .wrong = 0xFA0000C0, - }; - colors->line = (struct swaylock_colorset){ - .input = 0x000000FF, - .cleared = 0x000000FF, - .caps_lock = 0x000000FF, - .verifying = 0x000000FF, - .wrong = 0x000000FF, - }; - colors->ring = (struct swaylock_colorset){ - .input = 0x337D00FF, - .cleared = 0xE5A445FF, - .caps_lock = 0xE5A445FF, - .verifying = 0x3300FFFF, - .wrong = 0x7D3300FF, - }; - colors->text = (struct swaylock_colorset){ - .input = 0xE5A445FF, - .cleared = 0x000000FF, - .caps_lock = 0xE5A445FF, - .verifying = 0x000000FF, - .wrong = 0x000000FF, - }; -} - -enum line_mode { - LM_LINE, - LM_INSIDE, - LM_RING, -}; - -static int parse_options(int argc, char **argv, struct swaylock_state *state, - enum line_mode *line_mode, char **config_path) { - enum long_option_codes { - LO_BS_HL_COLOR = 256, - LO_CAPS_LOCK_BS_HL_COLOR, - LO_CAPS_LOCK_KEY_HL_COLOR, - LO_FONT, - LO_IND_RADIUS, - LO_IND_THICKNESS, - LO_INSIDE_COLOR, - LO_INSIDE_CLEAR_COLOR, - LO_INSIDE_CAPS_LOCK_COLOR, - LO_INSIDE_VER_COLOR, - LO_INSIDE_WRONG_COLOR, - LO_KEY_HL_COLOR, - LO_LINE_COLOR, - LO_LINE_CLEAR_COLOR, - LO_LINE_CAPS_LOCK_COLOR, - LO_LINE_VER_COLOR, - LO_LINE_WRONG_COLOR, - LO_RING_COLOR, - LO_RING_CLEAR_COLOR, - LO_RING_CAPS_LOCK_COLOR, - LO_RING_VER_COLOR, - LO_RING_WRONG_COLOR, - LO_SEP_COLOR, - LO_TEXT_COLOR, - LO_TEXT_CLEAR_COLOR, - LO_TEXT_CAPS_LOCK_COLOR, - LO_TEXT_VER_COLOR, - LO_TEXT_WRONG_COLOR, - }; - - static struct option long_options[] = { - {"config", required_argument, NULL, 'C'}, - {"color", required_argument, NULL, 'c'}, - {"ignore-empty-password", no_argument, NULL, 'e'}, - {"daemonize", no_argument, NULL, 'f'}, - {"help", no_argument, NULL, 'h'}, - {"image", required_argument, NULL, 'i'}, - {"disable-caps-lock-text", no_argument, NULL, 'L'}, - {"indicator-caps-lock", no_argument, NULL, 'l'}, - {"line-uses-inside", no_argument, NULL, 'n'}, - {"socket", required_argument, NULL, 'p'}, - {"line-uses-ring", no_argument, NULL, 'r'}, - {"scaling", required_argument, NULL, 's'}, - {"tiling", no_argument, NULL, 't'}, - {"no-unlock-indicator", no_argument, NULL, 'u'}, - {"version", no_argument, NULL, 'v'}, - {"bs-hl-color", required_argument, NULL, LO_BS_HL_COLOR}, - {"caps-lock-bs-hl-color", required_argument, NULL, LO_CAPS_LOCK_BS_HL_COLOR}, - {"caps-lock-key-hl-color", required_argument, NULL, LO_CAPS_LOCK_KEY_HL_COLOR}, - {"font", required_argument, NULL, LO_FONT}, - {"indicator-radius", required_argument, NULL, LO_IND_RADIUS}, - {"indicator-thickness", required_argument, NULL, LO_IND_THICKNESS}, - {"inside-color", required_argument, NULL, LO_INSIDE_COLOR}, - {"inside-clear-color", required_argument, NULL, LO_INSIDE_CLEAR_COLOR}, - {"inside-caps-lock-color", required_argument, NULL, LO_INSIDE_CAPS_LOCK_COLOR}, - {"inside-ver-color", required_argument, NULL, LO_INSIDE_VER_COLOR}, - {"inside-wrong-color", required_argument, NULL, LO_INSIDE_WRONG_COLOR}, - {"key-hl-color", required_argument, NULL, LO_KEY_HL_COLOR}, - {"line-color", required_argument, NULL, LO_LINE_COLOR}, - {"line-clear-color", required_argument, NULL, LO_LINE_CLEAR_COLOR}, - {"line-caps-lock-color", required_argument, NULL, LO_LINE_CAPS_LOCK_COLOR}, - {"line-ver-color", required_argument, NULL, LO_LINE_VER_COLOR}, - {"line-wrong-color", required_argument, NULL, LO_LINE_WRONG_COLOR}, - {"ring-color", required_argument, NULL, LO_RING_COLOR}, - {"ring-clear-color", required_argument, NULL, LO_RING_CLEAR_COLOR}, - {"ring-caps-lock-color", required_argument, NULL, LO_RING_CAPS_LOCK_COLOR}, - {"ring-ver-color", required_argument, NULL, LO_RING_VER_COLOR}, - {"ring-wrong-color", required_argument, NULL, LO_RING_WRONG_COLOR}, - {"separator-color", required_argument, NULL, LO_SEP_COLOR}, - {"text-color", required_argument, NULL, LO_TEXT_COLOR}, - {"text-clear-color", required_argument, NULL, LO_TEXT_CLEAR_COLOR}, - {"text-caps-lock-color", required_argument, NULL, LO_TEXT_CAPS_LOCK_COLOR}, - {"text-ver-color", required_argument, NULL, LO_TEXT_VER_COLOR}, - {"text-wrong-color", required_argument, NULL, LO_TEXT_WRONG_COLOR}, - {0, 0, 0, 0} - }; - - const char usage[] = - "Usage: swaylock [options...]\n" - "\n" - " -C, --config <config_file> " - "Path to the config file.\n" - " -c, --color <color> " - "Turn the screen into the given color instead of white.\n" - " -e, --ignore-empty-password " - "When an empty password is provided, do not validate it.\n" - " -f, --daemonize " - "Detach from the controlling terminal after locking.\n" - " -h, --help " - "Show help message and quit.\n" - " -i, --image [<output>:]<path> " - "Display the given image.\n" - " -L, --disable-caps-lock-text " - "Disable the Caps Lock text.\n" - " -l, --indicator-caps-lock " - "Show the current Caps Lock state also on the indicator.\n" - " -s, --scaling <mode> " - "Scaling mode: stretch, fill, fit, center, tile.\n" - " -t, --tiling " - "Same as --scaling=tile.\n" - " -u, --no-unlock-indicator " - "Disable the unlock indicator.\n" - " -v, --version " - "Show the version number and quit.\n" - " --bs-hl-color <color> " - "Sets the color of backspace highlight segments.\n" - " --caps-lock-bs-hl-color <color> " - "Sets the color of backspace highlight segments when Caps Lock " - "is active.\n" - " --caps-lock-key-hl-color <color> " - "Sets the color of the key press highlight segments when " - "Caps Lock is active.\n" - " --font <font> " - "Sets the font of the text.\n" - " --indicator-radius <radius> " - "Sets the indicator radius.\n" - " --indicator-thickness <thick> " - "Sets the indicator thickness.\n" - " --inside-color <color> " - "Sets the color of the inside of the indicator.\n" - " --inside-clear-color <color> " - "Sets the color of the inside of the indicator when cleared.\n" - " --inside-caps-lock-color <color> " - "Sets the color of the inside of the indicator when Caps Lock " - "is active.\n" - " --inside-ver-color <color> " - "Sets the color of the inside of the indicator when verifying.\n" - " --inside-wrong-color <color> " - "Sets the color of the inside of the indicator when invalid.\n" - " --key-hl-color <color> " - "Sets the color of the key press highlight segments.\n" - " --line-color <color> " - "Sets the color of the line between the inside and ring.\n" - " --line-clear-color <color> " - "Sets the color of the line between the inside and ring when " - "cleared.\n" - " --line-caps-lock-color <color> " - "Sets the color of the line between the inside and ring when " - "Caps Lock is active.\n" - " --line-ver-color <color> " - "Sets the color of the line between the inside and ring when " - "verifying.\n" - " --line-wrong-color <color> " - "Sets the color of the line between the inside and ring when " - "invalid.\n" - " -n, --line-uses-inside " - "Use the inside color for the line between the inside and ring.\n" - " -r, --line-uses-ring " - "Use the ring color for the line between the inside and ring.\n" - " --ring-color <color> " - "Sets the color of the ring of the indicator.\n" - " --ring-clear-color <color> " - "Sets the color of the ring of the indicator when cleared.\n" - " --ring-caps-lock-color <color> " - "Sets the color of the ring of the indicator when Caps Lock " - "is active.\n" - " --ring-ver-color <color> " - "Sets the color of the ring of the indicator when verifying.\n" - " --ring-wrong-color <color> " - "Sets the color of the ring of the indicator when invalid.\n" - " --separator-color <color> " - "Sets the color of the lines that separate highlight segments.\n" - " --text-color <color> " - "Sets the color of the text.\n" - " --text-clear-color <color> " - "Sets the color of the text when cleared.\n" - " --text-caps-lock-color <color> " - "Sets the color of the text when Caps Lock is active.\n" - " --text-ver-color <color> " - "Sets the color of the text when verifying.\n" - " --text-wrong-color <color> " - "Sets the color of the text when invalid.\n" - "\n" - "All <color> options are of the form <rrggbb[aa]>.\n"; - - int c; - optind = 1; - while (1) { - int opt_idx = 0; - c = getopt_long(argc, argv, "c:efhi:Llnrs:tuvC:", long_options, &opt_idx); - if (c == -1) { - break; - } - switch (c) { - case 'C': - if (config_path) { - *config_path = strdup(optarg); - } - break; - case 'c': - if (state) { - state->args.colors.background = parse_color(optarg); - state->args.mode = BACKGROUND_MODE_SOLID_COLOR; - } - break; - case 'e': - if (state) { - state->args.ignore_empty = true; - } - break; - case 'f': - if (state) { - state->args.daemonize = true; - } - break; - case 'i': - if (state) { - load_image(optarg, state); - } - break; - case 'L': - if (state) { - state->args.show_caps_lock_text = false; - } - break; - case 'l': - if (state) { - state->args.show_caps_lock_indicator = true; - } - break; - case 'n': - if (line_mode) { - *line_mode = LM_INSIDE; - } - break; - case 'r': - if (line_mode) { - *line_mode = LM_RING; - } - break; - case 's': - if (state) { - state->args.mode = parse_background_mode(optarg); - if (state->args.mode == BACKGROUND_MODE_INVALID) { - return 1; - } - } - break; - case 't': - if (state) { - state->args.mode = BACKGROUND_MODE_TILE; - } - break; - case 'u': - if (state) { - state->args.show_indicator = false; - } - break; - case 'v': - fprintf(stdout, "swaylock version " SWAY_VERSION "\n"); - exit(EXIT_SUCCESS); - break; - case LO_BS_HL_COLOR: - if (state) { - state->args.colors.bs_highlight = parse_color(optarg); - } - break; - case LO_CAPS_LOCK_BS_HL_COLOR: - if (state) { - state->args.colors.caps_lock_bs_highlight = parse_color(optarg); - } - break; - case LO_CAPS_LOCK_KEY_HL_COLOR: - if (state) { - state->args.colors.caps_lock_key_highlight = parse_color(optarg); - } - break; - case LO_FONT: - if (state) { - free(state->args.font); - state->args.font = strdup(optarg); - } - break; - case LO_IND_RADIUS: - if (state) { - state->args.radius = strtol(optarg, NULL, 0); - } - break; - case LO_IND_THICKNESS: - if (state) { - state->args.thickness = strtol(optarg, NULL, 0); - } - break; - case LO_INSIDE_COLOR: - if (state) { - state->args.colors.inside.input = parse_color(optarg); - } - break; - case LO_INSIDE_CLEAR_COLOR: - if (state) { - state->args.colors.inside.cleared = parse_color(optarg); - } - break; - case LO_INSIDE_CAPS_LOCK_COLOR: - if (state) { - state->args.colors.inside.caps_lock = parse_color(optarg); - } - break; - case LO_INSIDE_VER_COLOR: - if (state) { - state->args.colors.inside.verifying = parse_color(optarg); - } - break; - case LO_INSIDE_WRONG_COLOR: - if (state) { - state->args.colors.inside.wrong = parse_color(optarg); - } - break; - case LO_KEY_HL_COLOR: - if (state) { - state->args.colors.key_highlight = parse_color(optarg); - } - break; - case LO_LINE_COLOR: - if (state) { - state->args.colors.line.input = parse_color(optarg); - } - break; - case LO_LINE_CLEAR_COLOR: - if (state) { - state->args.colors.line.cleared = parse_color(optarg); - } - break; - case LO_LINE_CAPS_LOCK_COLOR: - if (state) { - state->args.colors.line.caps_lock = parse_color(optarg); - } - break; - case LO_LINE_VER_COLOR: - if (state) { - state->args.colors.line.verifying = parse_color(optarg); - } - break; - case LO_LINE_WRONG_COLOR: - if (state) { - state->args.colors.line.wrong = parse_color(optarg); - } - break; - case LO_RING_COLOR: - if (state) { - state->args.colors.ring.input = parse_color(optarg); - } - break; - case LO_RING_CLEAR_COLOR: - if (state) { - state->args.colors.ring.cleared = parse_color(optarg); - } - break; - case LO_RING_CAPS_LOCK_COLOR: - if (state) { - state->args.colors.ring.caps_lock = parse_color(optarg); - } - break; - case LO_RING_VER_COLOR: - if (state) { - state->args.colors.ring.verifying = parse_color(optarg); - } - break; - case LO_RING_WRONG_COLOR: - if (state) { - state->args.colors.ring.wrong = parse_color(optarg); - } - break; - case LO_SEP_COLOR: - if (state) { - state->args.colors.separator = parse_color(optarg); - } - break; - case LO_TEXT_COLOR: - if (state) { - state->args.colors.text.input = parse_color(optarg); - } - break; - case LO_TEXT_CLEAR_COLOR: - if (state) { - state->args.colors.text.cleared = parse_color(optarg); - } - break; - case LO_TEXT_CAPS_LOCK_COLOR: - if (state) { - state->args.colors.text.caps_lock = parse_color(optarg); - } - break; - case LO_TEXT_VER_COLOR: - if (state) { - state->args.colors.text.verifying = parse_color(optarg); - } - break; - case LO_TEXT_WRONG_COLOR: - if (state) { - state->args.colors.text.wrong = parse_color(optarg); - } - break; - default: - fprintf(stderr, "%s", usage); - return 1; - } - } - - return 0; -} - -static bool file_exists(const char *path) { - return path && access(path, R_OK) != -1; -} - -static char *get_config_path(void) { - static const char *config_paths[] = { - "$HOME/.swaylock/config", - "$XDG_CONFIG_HOME/swaylock/config", - SYSCONFDIR "/swaylock/config", - }; - - if (!getenv("XDG_CONFIG_HOME")) { - char *home = getenv("HOME"); - char *config_home = malloc(strlen(home) + strlen("/.config") + 1); - if (!config_home) { - wlr_log(WLR_ERROR, "Unable to allocate $HOME/.config"); - } else { - strcpy(config_home, home); - strcat(config_home, "/.config"); - setenv("XDG_CONFIG_HOME", config_home, 1); - wlr_log(WLR_DEBUG, "Set XDG_CONFIG_HOME to %s", config_home); - free(config_home); - } - } - - wordexp_t p; - char *path; - for (size_t i = 0; i < sizeof(config_paths) / sizeof(char *); ++i) { - if (wordexp(config_paths[i], &p, 0) == 0) { - path = strdup(p.we_wordv[0]); - wordfree(&p); - if (file_exists(path)) { - return path; - } - free(path); - } - } - - return NULL; -} - -static int load_config(char *path, struct swaylock_state *state, - enum line_mode *line_mode) { - FILE *config = fopen(path, "r"); - if (!config) { - wlr_log(WLR_ERROR, "Failed to read config. Running without it."); - return 0; - } - char *line = NULL; - size_t line_size = 0; - ssize_t nread; - int line_number = 0; - int result = 0; - while ((nread = getline(&line, &line_size, config)) != -1) { - line_number++; - - if (line[nread - 1] == '\n') { - line[--nread] = '\0'; - } - - if (!*line || line[0] == '#') { - continue; - } - - wlr_log(WLR_DEBUG, "Config Line #%d: %s", line_number, line); - char flag[nread + 3]; - sprintf(flag, "--%s", line); - char *argv[] = {"swaylock", flag}; - result = parse_options(2, argv, state, line_mode, NULL); - if (result != 0) { - break; - } - } - free(line); - fclose(config); - return 0; -} - -static struct swaylock_state state; - -static void display_in(int fd, short mask, void *data) { - if (wl_display_dispatch(state.display) == -1) { - state.run_display = false; - } -} - -int main(int argc, char **argv) { - wlr_log_init(WLR_DEBUG, NULL); - initialize_pw_backend(); - - enum line_mode line_mode = LM_LINE; - state.args = (struct swaylock_args){ - .mode = BACKGROUND_MODE_FILL, - .font = strdup("sans-serif"), - .radius = 50, - .thickness = 10, - .ignore_empty = false, - .show_indicator = true, - .show_caps_lock_indicator = false, - .show_caps_lock_text = true - }; - wl_list_init(&state.images); - set_default_colors(&state.args.colors); - - char *config_path = NULL; - int result = parse_options(argc, argv, NULL, NULL, &config_path); - if (result != 0) { - free(config_path); - return result; - } - if (!config_path) { - config_path = get_config_path(); - } - - if (config_path) { - wlr_log(WLR_DEBUG, "Found config at %s", config_path); - int config_status = load_config(config_path, &state, &line_mode); - free(config_path); - if (config_status != 0) { - return config_status; - } - } - - if (argc > 1) { - wlr_log(WLR_DEBUG, "Parsing CLI Args"); - int result = parse_options(argc, argv, &state, &line_mode, NULL); - if (result != 0) { - return result; - } - } - - if (line_mode == LM_INSIDE) { - state.args.colors.line = state.args.colors.inside; - } else if (line_mode == LM_RING) { - state.args.colors.line = state.args.colors.ring; - } - -#ifdef __linux__ - // Most non-linux platforms require root to mlock() - if (mlock(state.password.buffer, sizeof(state.password.buffer)) != 0) { - sway_abort("Unable to mlock() password memory."); - } -#endif - - wl_list_init(&state.surfaces); - state.xkb.context = xkb_context_new(XKB_CONTEXT_NO_FLAGS); - state.display = wl_display_connect(NULL); - if (!state.display) { - sway_abort("Unable to connect to the compositor. " - "If your compositor is running, check or set the " - "WAYLAND_DISPLAY environment variable."); - } - - struct wl_registry *registry = wl_display_get_registry(state.display); - wl_registry_add_listener(registry, ®istry_listener, &state); - wl_display_roundtrip(state.display); - assert(state.compositor && state.layer_shell && state.shm); - if (!state.input_inhibit_manager) { - wlr_log(WLR_ERROR, "Compositor does not support the input inhibitor " - "protocol, refusing to run insecurely"); - return 1; - } - - if (wl_list_empty(&state.surfaces)) { - wlr_log(WLR_DEBUG, "Exiting - no outputs to show on."); - return 0; - } - - zwlr_input_inhibit_manager_v1_get_inhibitor(state.input_inhibit_manager); - if (wl_display_roundtrip(state.display) == -1) { - wlr_log(WLR_ERROR, "Exiting - failed to inhibit input:" - " is another lockscreen already running?"); - return 2; - } - - if (state.zxdg_output_manager) { - struct swaylock_surface *surface; - wl_list_for_each(surface, &state.surfaces, link) { - surface->xdg_output = zxdg_output_manager_v1_get_xdg_output( - state.zxdg_output_manager, surface->output); - zxdg_output_v1_add_listener( - surface->xdg_output, &_xdg_output_listener, surface); - } - wl_display_roundtrip(state.display); - } else { - wlr_log(WLR_INFO, "Compositor does not support zxdg output manager, " - "images assigned to named outputs will not work"); - } - - struct swaylock_surface *surface; - wl_list_for_each(surface, &state.surfaces, link) { - create_layer_surface(surface); - } - - if (state.args.daemonize) { - wl_display_roundtrip(state.display); - daemonize(); - } - - state.eventloop = loop_create(); - loop_add_fd(state.eventloop, wl_display_get_fd(state.display), POLLIN, - display_in, NULL); - - state.run_display = true; - while (state.run_display) { - errno = 0; - if (wl_display_flush(state.display) == -1 && errno != EAGAIN) { - break; - } - loop_poll(state.eventloop); - } - - free(state.args.font); - return 0; -} diff --git a/swaylock/meson.build b/swaylock/meson.build deleted file mode 100644 index f3321a78..00000000 --- a/swaylock/meson.build +++ /dev/null @@ -1,55 +0,0 @@ -sysconfdir = get_option('sysconfdir') - -dependencies = [ - cairo, - client_protos, - gdk_pixbuf, - math, - pango, - pangocairo, - xkbcommon, - wayland_client, - wlroots, -] - -sources = [ - 'main.c', - 'password.c', - 'render.c', - 'seat.c' -] - -if libpam.found() - sources += ['pam.c'] - dependencies += [libpam] -else - warning('The swaylock binary must be setuid when compiled without libpam') - warning('You must do this manually post-install: chmod a+s /path/to/swaylock') - sources += ['shadow.c'] - if crypt.found() - dependencies += [crypt] - endif -endif - -executable('swaylock', - sources, - include_directories: [sway_inc], - dependencies: dependencies, - link_with: [lib_sway_common, lib_sway_client], - install_rpath : rpathdir, - install: true -) - -if is_freebsd - install_data( - 'pam/swaylock.freebsd', - install_dir: sysconfdir + '/pam.d/', - rename: 'swaylock' - ) -else - install_data( - 'pam/swaylock.linux', - install_dir: sysconfdir + '/pam.d/', - rename: 'swaylock' - ) -endif diff --git a/swaylock/pam.c b/swaylock/pam.c deleted file mode 100644 index b90d9e87..00000000 --- a/swaylock/pam.c +++ /dev/null @@ -1,62 +0,0 @@ -#define _POSIX_C_SOURCE 200809L -#include <pwd.h> -#include <security/pam_appl.h> -#include <stdbool.h> -#include <stdlib.h> -#include <string.h> -#include <unistd.h> -#include <wlr/util/log.h> -#include "swaylock/swaylock.h" - -void initialize_pw_backend(void) { - // TODO: only call pam_start once. keep the same handle the whole time -} - -static int function_conversation(int num_msg, const struct pam_message **msg, - struct pam_response **resp, void *data) { - struct swaylock_password *pw = data; - /* PAM expects an array of responses, one for each message */ - struct pam_response *pam_reply = calloc( - num_msg, sizeof(struct pam_response)); - *resp = pam_reply; - for (int i = 0; i < num_msg; ++i) { - switch (msg[i]->msg_style) { - case PAM_PROMPT_ECHO_OFF: - case PAM_PROMPT_ECHO_ON: - pam_reply[i].resp = strdup(pw->buffer); // PAM clears and frees this - break; - case PAM_ERROR_MSG: - case PAM_TEXT_INFO: - break; - } - } - return PAM_SUCCESS; -} - -bool attempt_password(struct swaylock_password *pw) { - struct passwd *passwd = getpwuid(getuid()); - char *username = passwd->pw_name; - const struct pam_conv local_conversation = { - function_conversation, pw - }; - pam_handle_t *local_auth_handle = NULL; - int pam_err; - if ((pam_err = pam_start("swaylock", username, - &local_conversation, &local_auth_handle)) != PAM_SUCCESS) { - wlr_log(WLR_ERROR, "PAM returned error %d", pam_err); - } - if ((pam_err = pam_authenticate(local_auth_handle, 0)) != PAM_SUCCESS) { - wlr_log(WLR_ERROR, "pam_authenticate failed"); - goto fail; - } - // TODO: only call pam_end once we succeed at authing. refresh tokens beforehand - if ((pam_err = pam_end(local_auth_handle, pam_err)) != PAM_SUCCESS) { - wlr_log(WLR_ERROR, "pam_end failed"); - goto fail; - } - clear_password_buffer(pw); - return true; -fail: - clear_password_buffer(pw); - return false; -} diff --git a/swaylock/pam/swaylock.freebsd b/swaylock/pam/swaylock.freebsd deleted file mode 100644 index 603fc185..00000000 --- a/swaylock/pam/swaylock.freebsd +++ /dev/null @@ -1,6 +0,0 @@ -# -# PAM configuration file for the swaylock screen locker. By default, it includes -# the 'passwd' configuration file (see /etc/pam.d/passwd) -# - -auth include passwd diff --git a/swaylock/pam/swaylock.linux b/swaylock/pam/swaylock.linux deleted file mode 100644 index 6a36b0d6..00000000 --- a/swaylock/pam/swaylock.linux +++ /dev/null @@ -1,6 +0,0 @@ -# -# PAM configuration file for the swaylock screen locker. By default, it includes -# the 'login' configuration file (see /etc/pam.d/login) -# - -auth include login diff --git a/swaylock/password.c b/swaylock/password.c deleted file mode 100644 index 3bd113ad..00000000 --- a/swaylock/password.c +++ /dev/null @@ -1,190 +0,0 @@ -#include <assert.h> -#include <errno.h> -#include <pwd.h> -#include <stdlib.h> -#include <string.h> -#include <unistd.h> -#include <wlr/util/log.h> -#include <xkbcommon/xkbcommon.h> -#include "swaylock/swaylock.h" -#include "swaylock/seat.h" -#include "loop.h" -#include "unicode.h" - -void clear_password_buffer(struct swaylock_password *pw) { - // Use volatile keyword so so compiler can't optimize this out. - volatile char *buffer = pw->buffer; - volatile char zero = '\0'; - for (size_t i = 0; i < sizeof(pw->buffer); ++i) { - buffer[i] = zero; - } - pw->len = 0; -} - -static bool backspace(struct swaylock_password *pw) { - if (pw->len != 0) { - pw->buffer[--pw->len] = 0; - return true; - } - return false; -} - -static void append_ch(struct swaylock_password *pw, uint32_t codepoint) { - size_t utf8_size = utf8_chsize(codepoint); - if (pw->len + utf8_size + 1 >= sizeof(pw->buffer)) { - // TODO: Display error - return; - } - utf8_encode(&pw->buffer[pw->len], codepoint); - pw->buffer[pw->len + utf8_size] = 0; - pw->len += utf8_size; -} - -static void clear_indicator(void *data) { - struct swaylock_state *state = data; - state->clear_indicator_timer = NULL; - state->auth_state = AUTH_STATE_IDLE; - damage_state(state); -} - -static void schedule_indicator_clear(struct swaylock_state *state) { - if (state->clear_indicator_timer) { - loop_remove_timer(state->eventloop, state->clear_indicator_timer); - } - state->clear_indicator_timer = loop_add_timer( - state->eventloop, 3000, clear_indicator, state); -} - -static void clear_password(void *data) { - struct swaylock_state *state = data; - state->clear_password_timer = NULL; - state->auth_state = AUTH_STATE_CLEAR; - clear_password_buffer(&state->password); - damage_state(state); - schedule_indicator_clear(state); -} - -static void schedule_password_clear(struct swaylock_state *state) { - if (state->clear_password_timer) { - loop_remove_timer(state->eventloop, state->clear_password_timer); - } - state->clear_password_timer = loop_add_timer( - state->eventloop, 10000, clear_password, state); -} - -static void handle_preverify_timeout(void *data) { - struct swaylock_state *state = data; - state->verify_password_timer = NULL; -} - -static void submit_password(struct swaylock_state *state) { - if (state->args.ignore_empty && state->password.len == 0) { - return; - } - - state->auth_state = AUTH_STATE_VALIDATING; - damage_state(state); - - // We generally want to wait until all surfaces are showing the - // "verifying" state before we go and verify the password, because - // verifying it is a blocking operation. However, if the surface is on - // an output with DPMS off then it won't update, so we set a timer. - state->verify_password_timer = loop_add_timer( - state->eventloop, 50, handle_preverify_timeout, state); - - while (state->run_display && state->verify_password_timer) { - errno = 0; - if (wl_display_flush(state->display) == -1 && errno != EAGAIN) { - break; - } - loop_poll(state->eventloop); - - bool ok = 1; - struct swaylock_surface *surface; - wl_list_for_each(surface, &state->surfaces, link) { - if (surface->dirty) { - ok = 0; - } - } - if (ok) { - break; - } - } - wl_display_flush(state->display); - - if (attempt_password(&state->password)) { - state->run_display = false; - return; - } - state->auth_state = AUTH_STATE_INVALID; - damage_state(state); - schedule_indicator_clear(state); -} - -void swaylock_handle_key(struct swaylock_state *state, - xkb_keysym_t keysym, uint32_t codepoint) { - switch (keysym) { - case XKB_KEY_KP_Enter: /* fallthrough */ - case XKB_KEY_Return: - submit_password(state); - break; - case XKB_KEY_Delete: - case XKB_KEY_BackSpace: - if (backspace(&state->password)) { - state->auth_state = AUTH_STATE_BACKSPACE; - } else { - state->auth_state = AUTH_STATE_CLEAR; - } - damage_state(state); - schedule_indicator_clear(state); - schedule_password_clear(state); - break; - case XKB_KEY_Escape: - clear_password_buffer(&state->password); - state->auth_state = AUTH_STATE_CLEAR; - damage_state(state); - schedule_indicator_clear(state); - break; - case XKB_KEY_Caps_Lock: - case XKB_KEY_Shift_L: - case XKB_KEY_Shift_R: - case XKB_KEY_Control_L: - case XKB_KEY_Control_R: - case XKB_KEY_Meta_L: - case XKB_KEY_Meta_R: - case XKB_KEY_Alt_L: - case XKB_KEY_Alt_R: - case XKB_KEY_Super_L: - case XKB_KEY_Super_R: - state->auth_state = AUTH_STATE_INPUT_NOP; - damage_state(state); - schedule_indicator_clear(state); - schedule_password_clear(state); - break; - case XKB_KEY_d: - if (state->xkb.control) { - submit_password(state); - break; - } - // fallthrough - case XKB_KEY_c: /* fallthrough */ - case XKB_KEY_u: - if (state->xkb.control) { - clear_password_buffer(&state->password); - state->auth_state = AUTH_STATE_CLEAR; - damage_state(state); - schedule_indicator_clear(state); - break; - } - // fallthrough - default: - if (codepoint) { - append_ch(&state->password, codepoint); - state->auth_state = AUTH_STATE_INPUT; - damage_state(state); - schedule_indicator_clear(state); - schedule_password_clear(state); - } - break; - } -} diff --git a/swaylock/render.c b/swaylock/render.c deleted file mode 100644 index 5aedaad5..00000000 --- a/swaylock/render.c +++ /dev/null @@ -1,189 +0,0 @@ -#include <math.h> -#include <stdlib.h> -#include <wayland-client.h> -#include "cairo.h" -#include "background-image.h" -#include "swaylock/swaylock.h" - -#define M_PI 3.14159265358979323846 -const float TYPE_INDICATOR_RANGE = M_PI / 3.0f; -const float TYPE_INDICATOR_BORDER_THICKNESS = M_PI / 128.0f; - -static void set_color_for_state(cairo_t *cairo, struct swaylock_state *state, - struct swaylock_colorset *colorset) { - if (state->auth_state == AUTH_STATE_VALIDATING) { - cairo_set_source_u32(cairo, colorset->verifying); - } else if (state->auth_state == AUTH_STATE_INVALID) { - cairo_set_source_u32(cairo, colorset->wrong); - } else if (state->auth_state == AUTH_STATE_CLEAR) { - cairo_set_source_u32(cairo, colorset->cleared); - } else { - if (state->xkb.caps_lock && state->args.show_caps_lock_indicator) { - cairo_set_source_u32(cairo, colorset->caps_lock); - } else if (state->xkb.caps_lock && !state->args.show_caps_lock_indicator && - state->args.show_caps_lock_text) { - uint32_t inputtextcolor = state->args.colors.text.input; - state->args.colors.text.input = state->args.colors.text.caps_lock; - cairo_set_source_u32(cairo, colorset->input); - state->args.colors.text.input = inputtextcolor; - } else { - cairo_set_source_u32(cairo, colorset->input); - } - } -} - -void render_frame(struct swaylock_surface *surface) { - struct swaylock_state *state = surface->state; - - int buffer_width = surface->width * surface->scale; - int buffer_height = surface->height * surface->scale; - if (buffer_width == 0 || buffer_height == 0) { - return; // not yet configured - } - - surface->current_buffer = get_next_buffer(state->shm, - surface->buffers, buffer_width, buffer_height); - if (surface->current_buffer == NULL) { - return; - } - - cairo_t *cairo = surface->current_buffer->cairo; - cairo_set_antialias(cairo, CAIRO_ANTIALIAS_BEST); - cairo_font_options_t *fo = cairo_font_options_create(); - cairo_font_options_set_hint_style(fo, CAIRO_HINT_STYLE_FULL); - cairo_font_options_set_antialias(fo, CAIRO_ANTIALIAS_SUBPIXEL); - cairo_font_options_set_subpixel_order(fo, to_cairo_subpixel_order(surface->subpixel)); - cairo_set_font_options(cairo, fo); - cairo_font_options_destroy(fo); - cairo_identity_matrix(cairo); - - cairo_save(cairo); - cairo_set_operator(cairo, CAIRO_OPERATOR_SOURCE); - if (state->args.mode == BACKGROUND_MODE_SOLID_COLOR || !surface->image) { - cairo_set_source_u32(cairo, state->args.colors.background); - cairo_paint(cairo); - } else { - render_background_image(cairo, surface->image, - state->args.mode, buffer_width, buffer_height); - } - cairo_restore(cairo); - cairo_identity_matrix(cairo); - - int arc_radius = state->args.radius * surface->scale; - int arc_thickness = state->args.thickness * surface->scale; - float type_indicator_border_thickness = - TYPE_INDICATOR_BORDER_THICKNESS * surface->scale; - - if (state->args.show_indicator && state->auth_state != AUTH_STATE_IDLE) { - // Draw circle - cairo_set_line_width(cairo, arc_thickness); - cairo_arc(cairo, buffer_width / 2, buffer_height / 2, arc_radius, - 0, 2 * M_PI); - set_color_for_state(cairo, state, &state->args.colors.inside); - cairo_fill_preserve(cairo); - set_color_for_state(cairo, state, &state->args.colors.ring); - cairo_stroke(cairo); - - // Draw a message - char *text = NULL; - set_color_for_state(cairo, state, &state->args.colors.text); - cairo_select_font_face(cairo, state->args.font, - CAIRO_FONT_SLANT_NORMAL, CAIRO_FONT_WEIGHT_NORMAL); - cairo_set_font_size(cairo, arc_radius / 3.0f); - switch (state->auth_state) { - case AUTH_STATE_VALIDATING: - text = "verifying"; - break; - case AUTH_STATE_INVALID: - text = "wrong"; - break; - case AUTH_STATE_CLEAR: - text = "cleared"; - break; - case AUTH_STATE_INPUT: - case AUTH_STATE_INPUT_NOP: - case AUTH_STATE_BACKSPACE: - if (state->xkb.caps_lock && state->args.show_caps_lock_text) { - text = "Caps Lock"; - } - break; - default: - break; - } - - if (text) { - cairo_text_extents_t extents; - double x, y; - cairo_text_extents(cairo, text, &extents); - x = (buffer_width / 2) - - (extents.width / 2 + extents.x_bearing); - y = (buffer_height / 2) - - (extents.height / 2 + extents.y_bearing); - - cairo_move_to(cairo, x, y); - cairo_show_text(cairo, text); - cairo_close_path(cairo); - cairo_new_sub_path(cairo); - } - - // Typing indicator: Highlight random part on keypress - if (state->auth_state == AUTH_STATE_INPUT - || state->auth_state == AUTH_STATE_BACKSPACE) { - static double highlight_start = 0; - highlight_start += - (rand() % (int)(M_PI * 100)) / 100.0 + M_PI * 0.5; - cairo_arc(cairo, buffer_width / 2, buffer_height / 2, - arc_radius, highlight_start, - highlight_start + TYPE_INDICATOR_RANGE); - if (state->auth_state == AUTH_STATE_INPUT) { - if (state->xkb.caps_lock && state->args.show_caps_lock_indicator) { - cairo_set_source_u32(cairo, state->args.colors.caps_lock_key_highlight); - } else { - cairo_set_source_u32(cairo, state->args.colors.key_highlight); - } - } else { - if (state->xkb.caps_lock && state->args.show_caps_lock_indicator) { - cairo_set_source_u32(cairo, state->args.colors.caps_lock_bs_highlight); - } else { - cairo_set_source_u32(cairo, state->args.colors.bs_highlight); - } - } - cairo_stroke(cairo); - - // Draw borders - cairo_set_source_u32(cairo, state->args.colors.separator); - cairo_arc(cairo, buffer_width / 2, buffer_height / 2, - arc_radius, highlight_start, - highlight_start + type_indicator_border_thickness); - cairo_stroke(cairo); - - cairo_arc(cairo, buffer_width / 2, buffer_height / 2, - arc_radius, highlight_start + TYPE_INDICATOR_RANGE, - highlight_start + TYPE_INDICATOR_RANGE + - type_indicator_border_thickness); - cairo_stroke(cairo); - } - - // Draw inner + outer border of the circle - set_color_for_state(cairo, state, &state->args.colors.line); - cairo_set_line_width(cairo, 2.0 * surface->scale); - cairo_arc(cairo, buffer_width / 2, buffer_height / 2, - arc_radius - arc_thickness / 2, 0, 2 * M_PI); - cairo_stroke(cairo); - cairo_arc(cairo, buffer_width / 2, buffer_height / 2, - arc_radius + arc_thickness / 2, 0, 2 * M_PI); - cairo_stroke(cairo); - } - - wl_surface_set_buffer_scale(surface->surface, surface->scale); - wl_surface_attach(surface->surface, surface->current_buffer->buffer, 0, 0); - wl_surface_damage(surface->surface, 0, 0, surface->width, surface->height); - wl_surface_commit(surface->surface); -} - -void render_frames(struct swaylock_state *state) { - struct swaylock_surface *surface; - wl_list_for_each(surface, &state->surfaces, link) { - render_frame(surface); - } -} diff --git a/swaylock/seat.c b/swaylock/seat.c deleted file mode 100644 index f0b1385e..00000000 --- a/swaylock/seat.c +++ /dev/null @@ -1,178 +0,0 @@ -#include <assert.h> -#include <stdlib.h> -#include <sys/mman.h> -#include <unistd.h> -#include <wlr/util/log.h> -#include <xkbcommon/xkbcommon.h> -#include "swaylock/swaylock.h" -#include "swaylock/seat.h" - -static void keyboard_keymap(void *data, struct wl_keyboard *wl_keyboard, - uint32_t format, int32_t fd, uint32_t size) { - struct swaylock_state *state = data; - if (format != WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1) { - close(fd); - wlr_log(WLR_ERROR, "Unknown keymap format %d, aborting", format); - exit(1); - } - char *map_shm = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0); - if (map_shm == MAP_FAILED) { - close(fd); - wlr_log(WLR_ERROR, "Unable to initialize keymap shm, aborting"); - exit(1); - } - struct xkb_keymap *keymap = xkb_keymap_new_from_string( - state->xkb.context, map_shm, XKB_KEYMAP_FORMAT_TEXT_V1, 0); - munmap(map_shm, size); - close(fd); - assert(keymap); - struct xkb_state *xkb_state = xkb_state_new(keymap); - assert(xkb_state); - xkb_keymap_unref(state->xkb.keymap); - xkb_state_unref(state->xkb.state); - state->xkb.keymap = keymap; - state->xkb.state = xkb_state; -} - -static void keyboard_enter(void *data, struct wl_keyboard *wl_keyboard, - uint32_t serial, struct wl_surface *surface, struct wl_array *keys) { - // Who cares -} - -static void keyboard_leave(void *data, struct wl_keyboard *wl_keyboard, - uint32_t serial, struct wl_surface *surface) { - // Who cares -} - -static void keyboard_key(void *data, struct wl_keyboard *wl_keyboard, - uint32_t serial, uint32_t time, uint32_t key, uint32_t _key_state) { - struct swaylock_state *state = data; - enum wl_keyboard_key_state key_state = _key_state; - xkb_keysym_t sym = xkb_state_key_get_one_sym(state->xkb.state, key + 8); - uint32_t keycode = key_state == WL_KEYBOARD_KEY_STATE_PRESSED ? - key + 8 : 0; - uint32_t codepoint = xkb_state_key_get_utf32(state->xkb.state, keycode); - if (key_state == WL_KEYBOARD_KEY_STATE_PRESSED) { - swaylock_handle_key(state, sym, codepoint); - } -} - -static void keyboard_modifiers(void *data, struct wl_keyboard *wl_keyboard, - uint32_t serial, uint32_t mods_depressed, uint32_t mods_latched, - uint32_t mods_locked, uint32_t group) { - struct swaylock_state *state = data; - xkb_state_update_mask(state->xkb.state, - mods_depressed, mods_latched, mods_locked, 0, 0, group); - int caps_lock = xkb_state_mod_name_is_active(state->xkb.state, - XKB_MOD_NAME_CAPS, XKB_STATE_MODS_LOCKED); - if (caps_lock != state->xkb.caps_lock) { - state->xkb.caps_lock = caps_lock; - damage_state(state); - } - state->xkb.control = xkb_state_mod_name_is_active(state->xkb.state, - XKB_MOD_NAME_CTRL, - XKB_STATE_MODS_DEPRESSED | XKB_STATE_MODS_LATCHED); - -} - -static void keyboard_repeat_info(void *data, struct wl_keyboard *wl_keyboard, - int32_t rate, int32_t delay) { - // TODO -} - -static const struct wl_keyboard_listener keyboard_listener = { - .keymap = keyboard_keymap, - .enter = keyboard_enter, - .leave = keyboard_leave, - .key = keyboard_key, - .modifiers = keyboard_modifiers, - .repeat_info = keyboard_repeat_info, -}; - -static void wl_pointer_enter(void *data, struct wl_pointer *wl_pointer, - uint32_t serial, struct wl_surface *surface, - wl_fixed_t surface_x, wl_fixed_t surface_y) { - wl_pointer_set_cursor(wl_pointer, serial, NULL, 0, 0); -} - -static void wl_pointer_leave(void *data, struct wl_pointer *wl_pointer, - uint32_t serial, struct wl_surface *surface) { - // Who cares -} - -static void wl_pointer_motion(void *data, struct wl_pointer *wl_pointer, - uint32_t time, wl_fixed_t surface_x, wl_fixed_t surface_y) { - // Who cares -} - -static void wl_pointer_button(void *data, struct wl_pointer *wl_pointer, - uint32_t serial, uint32_t time, uint32_t button, uint32_t state) { - // Who cares -} - -static void wl_pointer_axis(void *data, struct wl_pointer *wl_pointer, - uint32_t time, uint32_t axis, wl_fixed_t value) { - // Who cares -} - -static void wl_pointer_frame(void *data, struct wl_pointer *wl_pointer) { - // Who cares -} - -static void wl_pointer_axis_source(void *data, struct wl_pointer *wl_pointer, - uint32_t axis_source) { - // Who cares -} - -static void wl_pointer_axis_stop(void *data, struct wl_pointer *wl_pointer, - uint32_t time, uint32_t axis) { - // Who cares -} - -static void wl_pointer_axis_discrete(void *data, struct wl_pointer *wl_pointer, - uint32_t axis, int32_t discrete) { - // Who cares -} - -static const struct wl_pointer_listener pointer_listener = { - .enter = wl_pointer_enter, - .leave = wl_pointer_leave, - .motion = wl_pointer_motion, - .button = wl_pointer_button, - .axis = wl_pointer_axis, - .frame = wl_pointer_frame, - .axis_source = wl_pointer_axis_source, - .axis_stop = wl_pointer_axis_stop, - .axis_discrete = wl_pointer_axis_discrete, -}; - -static void seat_handle_capabilities(void *data, struct wl_seat *wl_seat, - enum wl_seat_capability caps) { - struct swaylock_seat *seat = data; - if (seat->pointer) { - wl_pointer_release(seat->pointer); - seat->pointer = NULL; - } - if (seat->keyboard) { - wl_keyboard_release(seat->keyboard); - seat->keyboard = NULL; - } - if ((caps & WL_SEAT_CAPABILITY_POINTER)) { - seat->pointer = wl_seat_get_pointer(wl_seat); - wl_pointer_add_listener(seat->pointer, &pointer_listener, NULL); - } - if ((caps & WL_SEAT_CAPABILITY_KEYBOARD)) { - seat->keyboard = wl_seat_get_keyboard(wl_seat); - wl_keyboard_add_listener(seat->keyboard, &keyboard_listener, seat->state); - } -} - -static void seat_handle_name(void *data, struct wl_seat *wl_seat, - const char *name) { - // Who cares -} - -const struct wl_seat_listener seat_listener = { - .capabilities = seat_handle_capabilities, - .name = seat_handle_name, -}; diff --git a/swaylock/shadow.c b/swaylock/shadow.c deleted file mode 100644 index b7b10a67..00000000 --- a/swaylock/shadow.c +++ /dev/null @@ -1,155 +0,0 @@ -#define _XOPEN_SOURCE // for crypt -#include <pwd.h> -#include <shadow.h> -#include <stdbool.h> -#include <sys/types.h> -#include <unistd.h> -#include <wlr/util/log.h> -#include "swaylock/swaylock.h" -#ifdef __GLIBC__ -// GNU, you damn slimy bastard -#include <crypt.h> -#endif - -static int comm[2][2]; - -static void clear_buffer(void *buf, size_t bytes) { - volatile char *buffer = buf; - volatile char zero = '\0'; - for (size_t i = 0; i < bytes; ++i) { - buffer[i] = zero; - } -} - -void run_child(void) { - /* This code runs as root */ - struct passwd *pwent = getpwuid(getuid()); - if (!pwent) { - wlr_log_errno(WLR_ERROR, "failed to getpwuid"); - exit(EXIT_FAILURE); - } - char *encpw = pwent->pw_passwd; - if (strcmp(encpw, "x") == 0) { - struct spwd *swent = getspnam(pwent->pw_name); - if (!swent) { - wlr_log_errno(WLR_ERROR, "failed to getspnam"); - exit(EXIT_FAILURE); - } - encpw = swent->sp_pwdp; - } - - /* We don't need any additional logging here because the parent process will - * also fail here and will handle logging for us. */ - if (setgid(getgid()) != 0) { - exit(EXIT_FAILURE); - } - if (setuid(getuid()) != 0) { - exit(EXIT_FAILURE); - } - - /* This code does not run as root */ - wlr_log(WLR_DEBUG, "prepared to authorize user %s", pwent->pw_name); - - size_t size; - char *buf; - while (1) { - ssize_t amt; - amt = read(comm[0][0], &size, sizeof(size)); - if (amt == 0) { - break; - } else if (amt < 0) { - wlr_log_errno(WLR_ERROR, "read pw request"); - } - wlr_log(WLR_DEBUG, "received pw check request"); - buf = malloc(size); - if (!buf) { - wlr_log_errno(WLR_ERROR, "failed to malloc pw buffer"); - exit(EXIT_FAILURE); - } - size_t offs = 0; - do { - amt = read(comm[0][0], &buf[offs], size - offs); - if (amt <= 0) { - wlr_log_errno(WLR_ERROR, "failed to read pw"); - exit(EXIT_FAILURE); - } - offs += (size_t)amt; - } while (offs < size); - bool result = false; - char *c = crypt(buf, encpw); - if (c == NULL) { - wlr_log_errno(WLR_ERROR, "crypt"); - } - result = strcmp(c, encpw) == 0; - if (write(comm[1][1], &result, sizeof(result)) != sizeof(result)) { - wlr_log_errno(WLR_ERROR, "failed to write pw check result"); - clear_buffer(buf, size); - exit(EXIT_FAILURE); - } - clear_buffer(buf, size); - free(buf); - } - - clear_buffer(encpw, strlen(encpw)); - exit(EXIT_SUCCESS); -} - -void initialize_pw_backend(void) { - if (geteuid() != 0) { - wlr_log(WLR_ERROR, "swaylock needs to be setuid to read /etc/shadow"); - exit(EXIT_FAILURE); - } - if (pipe(comm[0]) != 0) { - wlr_log_errno(WLR_ERROR, "failed to create pipe"); - exit(EXIT_FAILURE); - } - if (pipe(comm[1]) != 0) { - wlr_log_errno(WLR_ERROR, "failed to create pipe"); - exit(EXIT_FAILURE); - } - pid_t child = fork(); - if (child == 0) { - close(comm[0][1]); - close(comm[1][0]); - run_child(); - } else if (child < 0) { - wlr_log_errno(WLR_ERROR, "failed to fork"); - exit(EXIT_FAILURE); - } - close(comm[0][0]); - close(comm[1][1]); - if (setgid(getgid()) != 0) { - wlr_log_errno(WLR_ERROR, "Unable to drop root"); - exit(EXIT_FAILURE); - } - if (setuid(getuid()) != 0) { - wlr_log_errno(WLR_ERROR, "Unable to drop root"); - exit(EXIT_FAILURE); - } -} - -bool attempt_password(struct swaylock_password *pw) { - bool result = false; - size_t len = pw->len + 1; - size_t offs = 0; - if (write(comm[0][1], &len, sizeof(len)) < 0) { - wlr_log_errno(WLR_ERROR, "Failed to request pw check"); - goto ret; - } - do { - ssize_t amt = write(comm[0][1], &pw->buffer[offs], len - offs); - if (amt < 0) { - wlr_log_errno(WLR_ERROR, "Failed to write pw buffer"); - goto ret; - } - offs += amt; - } while (offs < len); - if (read(comm[1][0], &result, sizeof(result)) != sizeof(result)) { - wlr_log_errno(WLR_ERROR, "Failed to read pw result"); - goto ret; - } - wlr_log(WLR_DEBUG, "pw result: %d", result); -ret: - clear_password_buffer(pw); - return result; -} diff --git a/swaylock/swaylock.1.scd b/swaylock/swaylock.1.scd deleted file mode 100644 index 2c7979be..00000000 --- a/swaylock/swaylock.1.scd +++ /dev/null @@ -1,167 +0,0 @@ -swaylock(1) - -# NAME - -swaylock - Screen locker for Wayland - -# SYNOPSIS - -_swaylock_ [options...] - -Locks your Wayland session. - -# OPTIONS - -*-C, --config* <path> - The config file to use. By default, the following paths are checked: - _$HOME/.swaylock/config_, _$XDG\_CONFIG\_HOME/swaylock/config_, and - _SYSCONFDIR/swaylock/config_. All flags aside from this one are valid - options in the configuration file using the format _long-option=value_. - For options such as _ignore-empty-password_, just supply the _long-option_. - All leading dashes should be omitted and the equals sign is required for - flags that take an argument. - -*-e, --ignore-empty-password* - When an empty password is provided by the user, do not validate it. - -*-f, --daemonize* - Detach from the controlling terminal after locking. - - Note: this is the default behavior of i3lock. - -*-h, --help* - Show help message and quit. - -*-v, --version* - Show the version number and quit. - -# APPEARANCE - -*-u, --no-unlock-indicator* - Disable the unlock indicator. - -*-i, --image* [<output>:]<path> - Display the given image, optionally only on the given output. Use -c to set - a background color. - -*-L, --disable-caps-lock-text* - Disable the Caps Lock Text. - -*-l, --indicator-caps-lock* - Show the current Caps Lock state also on the indicator. - -*-s, --scaling* - Scaling mode for images: _stretch_, _fill_, _fit_, _center_, or _tile_. - -*-t, --tiling* - Same as --scaling=tile. - -*-c, --color* <rrggbb[aa]> - Turn the screen into the given color. If -i is used, this sets the - background of the image to the given color. Defaults to white (FFFFFF), or - transparent (00000000) if an image is in use. - -*--bs-hl-color* <rrggbb[aa]> - Sets the color of backspace highlight segments. - -*--caps-lock-bs-hl-color* <rrggbb[aa]> - Sets the color of backspace highlight segments when Caps Lock is active. - -*--caps-lock-bs-hl-color* <rrggbb[aa]> - Sets the color of the key press highlight segments when Caps Lock is active. - -*--font* <font> - Sets the font of the text inside the indicator. - -*--indicator-radius* <radius> - Sets the radius of the indicator to _radius_ pixels. The default value is - 50. - -*--indicator-thickness* <thickness> - Sets the thickness of the indicator to _thickness_ pixels. The default value - is 10. - -*--inside-color* <rrggbb[aa]> - Sets the color of the inside of the indicator when typing or idle. - -*--inside-clear-color* <rrggbb[aa]> - Sets the color of the inside of the indicator when cleared. - -*--inside-caps-lock-color* <rrggbb[aa]> - Sets the color of the inside of the indicator when Caps Lock is active. - -*--inside-ver-color* <rrggbb[aa]> - Sets the color of the inside of the indicator when verifying. - -*--inside-wrong-color* <rrggbb[aa]> - Sets the color of the inside of the indicator when invalid. - -*--key-hl-color* <rrggbb[aa]> - Sets the color of key press highlight segments. - -*--line-color* <rrggbb[aa]> - Sets the color of the lines that separate the inside and outside of the - indicator when typing or idle. - -*--line-clear-color* <rrggbb[aa]> - Sets the color of the lines that separate the inside and outside of the - indicator when cleared. - -*--line-caps-lock-color* <rrggbb[aa]> - Sets the color of the line between the inside and ring when Caps Lock - is active. - -*--line-ver-color* <rrggbb[aa]> - Sets the color of the lines that separate the inside and outside of the - indicator when verifying. - -*--line-wrong-color* <rrggbb[aa]> - Sets the color of the lines that separate the inside and outside of the - indicator when invalid. - -*-n, --line-uses-inside* - Use the color of the inside of the indicator for the line separating the - inside and outside of the indicator. - -*-r, --line-uses-ring* - Use the outer ring's color for the line separating the inside and outside of - the indicator. - -*--ring-color* <rrggbb[aa]> - Sets the color of the outside of the indicator when typing or idle. - -*--ring-clear-color* <rrggbb[aa]> - Sets the color of the outside of the indicator when cleared. - -*--ring-caps-lock-color* <rrggbb[aa]> - Sets the color of the ring of the indicator when Caps Lock is active. - -*--ring-ver-color* <rrggbb[aa]> - Sets the color of the outside of the indicator when verifying. - -*--ring-wrong-color* <rrggbb[aa]> - Sets the color of the outside of the indicator when invalid. - -*--separator-color* <rrggbb[aa]> - Sets the color of the lines that separate highlight segments. - -*--text-color* <rrggbb[aa]> - Sets the color of the text inside the indicator when typing or idle. - -*--text-clear-color* <rrggbb[aa]> - Sets the color of the text inside the indicator when cleared. - -*--text-caps-lock-color* <rrggbb[aa]> - Sets the color of the text when Caps Lock is active. - -*--text-ver-color* <rrggbb[aa]> - Sets the color of the text inside the indicator when verifying. - -*--text-wrong-color* <rrggbb[aa]> - Sets the color of the text inside the indicator when invalid. - -# AUTHORS - -Maintained by Drew DeVault <sir@cmpwn.com>, who is assisted by other open -source contributors. For more information about sway development, see -https://github.com/swaywm/sway. |