#include "wayland-swaylock-client-protocol.h" #include #include #include #include #include #include #include #include #include #include "client/window.h" #include "client/registry.h" #include "client/cairo.h" #include "log.h" #include "list.h" list_t *surfaces; struct registry *registry; enum scaling_mode { SCALING_MODE_STRETCH, SCALING_MODE_FILL, SCALING_MODE_FIT, SCALING_MODE_CENTER, SCALING_MODE_TILE, }; enum none_one_list_choice { NOL_NONE, NOL_ONE, NOL_LIST }; struct none_one_list { enum none_one_list_choice type; void *a, *b; }; void sway_terminate(int exit_code) { int i; for (i = 0; i < surfaces->length; ++i) { struct window *window = surfaces->items[i]; window_teardown(window); } list_free(surfaces); registry_teardown(registry); exit(exit_code); } char *password; int password_size; int function_conversation(int num_msg, const struct pam_message **msg, struct pam_response **resp, void *appdata_ptr) { const char* msg_style_names[] = { NULL, "PAM_PROMPT_ECHO_OFF", "PAM_PROMPT_ECHO_ON", "PAM_ERROR_MSG", "PAM_TEXT_INFO", }; /* 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; imsg_style], msg[i]->msg); switch (msg[i]->msg_style) { case PAM_PROMPT_ECHO_OFF: case PAM_PROMPT_ECHO_ON: pam_reply[i].resp = password; break; case PAM_ERROR_MSG: case PAM_TEXT_INFO: break; } } return PAM_SUCCESS; } /** * password will be zeroed out. */ bool verify_password() { struct passwd *passwd = getpwuid(getuid()); char *username = passwd->pw_name; const struct pam_conv local_conversation = { function_conversation, NULL }; pam_handle_t *local_auth_handle = NULL; int pam_err; if ((pam_err = pam_start("swaylock", username, &local_conversation, &local_auth_handle)) != PAM_SUCCESS) { sway_abort("PAM returned %d\n", pam_err); } if ((pam_err = pam_authenticate(local_auth_handle, 0)) != PAM_SUCCESS) { return false; } if ((pam_err = pam_end(local_auth_handle, pam_err)) != PAM_SUCCESS) { return false; } return true; } void notify_key(enum wl_keyboard_key_state state, xkb_keysym_t sym, uint32_t code, uint32_t codepoint) { if (state == WL_KEYBOARD_KEY_STATE_PRESSED) { switch (sym) { case XKB_KEY_Return: if (verify_password()) { exit(0); } password_size = 1024; password = malloc(password_size); password[0] = '\0'; break; default: { int i = strlen(password); if (i + 1 == password_size) { password_size += 1024; password = realloc(password, password_size); } password[i] = (char)codepoint; password[i + 1] = '\0'; break; } } } } void render_color(struct window *window, uint32_t color) { cairo_set_source_u32(window->cairo, color); cairo_paint(window->cairo); } void render_image(struct window *window, cairo_surface_t *image, enum scaling_mode scaling_mode) { double width = cairo_image_surface_get_width(image); double height = cairo_image_surface_get_height(image); switch (scaling_mode) { case SCALING_MODE_STRETCH: cairo_scale(window->cairo, (double) window->width / width, (double) window->height / height); cairo_set_source_surface(window->cairo, image, 0, 0); break; case SCALING_MODE_FILL: { double window_ratio = (double) window->width / window->height; double bg_ratio = width / height; if (window_ratio > bg_ratio) { double scale = (double) window->width / width; cairo_scale(window->cairo, scale, scale); cairo_set_source_surface(window->cairo, image, 0, (double) window->height/2 / scale - height/2); } else { double scale = (double) window->height / height; cairo_scale(window->cairo, scale, scale); cairo_set_source_surface(window->cairo, image, (double) window->width/2 / scale - width/2, 0); } break; } case SCALING_MODE_FIT: { double window_ratio = (double) window->width / window->height; double bg_ratio = width / height; if (window_ratio > bg_ratio) { double scale = (double) window->height / height; cairo_scale(window->cairo, scale, scale); cairo_set_source_surface(window->cairo, image, (double) window->width/2 / scale - width/2, 0); } else { double scale = (double) window->width / width; cairo_scale(window->cairo, scale, scale); cairo_set_source_surface(window->cairo, image, 0, (double) window->height/2 / scale - height/2); } break; } case SCALING_MODE_CENTER: cairo_set_source_surface(window->cairo, image, (double) window->width/2 - width/2, (double) window->height/2 - height/2); break; case SCALING_MODE_TILE: { cairo_pattern_t *pattern = cairo_pattern_create_for_surface(image); cairo_pattern_set_extend(pattern, CAIRO_EXTEND_REPEAT); cairo_set_source(window->cairo, pattern); break; } } cairo_paint(window->cairo); } cairo_surface_t *load_image(char *image_path) { cairo_surface_t *image = NULL; #ifdef WITH_GDK_PIXBUF GError *err = NULL; GdkPixbuf *pixbuf = gdk_pixbuf_new_from_file(image_path, &err); if (!pixbuf) { fprintf(stderr, "GDK Error: %s\n", err->message); sway_abort("Failed to load background image."); } image = gdk_cairo_image_surface_create_from_pixbuf(pixbuf); g_object_unref(pixbuf); #else image = cairo_image_surface_create_from_png(image_path); #endif //WITH_GDK_PIXBUF if (!image) { sway_abort("Failed to read background image."); } return image; } int main(int argc, char **argv) { struct none_one_list images = { NOL_NONE, NULL, NULL }; char *scaling_mode_str = "fit"; uint32_t color = 0xFFFFFFFF; int i; init_log(L_INFO); static struct option long_options[] = { {"help", no_argument, NULL, 'h'}, {"color", required_argument, NULL, 'c'}, {"image", required_argument, NULL, 'i'}, {"scaling", required_argument, NULL, 's'}, {"tiling", no_argument, NULL, 't'}, {"version", no_argument, NULL, 'v'}, {"window-image", required_argument, NULL, 'w'}, {0, 0, 0, 0} }; const char *usage = "Usage: swaylock [options...]\n" "\n" " -h, --help Show help message and quit.\n" " -c, --color Turn the screen into the given color instead of white.\n" " -s, --scaling Scaling mode: stretch, fill, fit, center, tile.\n" " -t, --tiling Same as --scaling=tile.\n" " -v, --version Show the version number and quit.\n" " -i, --image Display the given image.\n" " --window-image : Display the given image on the given window.\n"; int c; while (1) { int option_index = 0; c = getopt_long(argc, argv, "hc:i:s:tv", long_options, &option_index); if (c == -1) { break; } switch (c) { case 'c': { int colorlen = strlen(optarg); if (colorlen < 6 || colorlen == 7 || colorlen > 8) { fprintf(stderr, "color must be specified in 3 or 4 byte format, e.g. ff0000 or ff0000ff\n"); exit(EXIT_FAILURE); } color = strtol(optarg, NULL, 16); if (colorlen == 6) { color <<= 8; color |= 0xFF; } sway_log(L_DEBUG, "color: 0x%x", color); break; } case 'i': if(images.type == NOL_NONE) { images = (struct none_one_list) { NOL_ONE, optarg, NULL }; } else { fprintf(stderr, "only one of --image/--window-image is allowed\n"); exit(EXIT_FAILURE); } break; case 'w': if(images.type == NOL_NONE) { images = (struct none_one_list) { NOL_LIST, create_list(), create_list() }; for(i = 0; i <= ((list_t*) images.a)->capacity; ++i) { // this isn't perfect - if you hav>e more than 10 monitors there's a small chance of // a segfault from uninitialized data. I honestly don't think enough people have more than // 10 monitors to care about that though. ((list_t*) images.a)->items[i] = NULL; ((list_t*) images.b)->items[i] = NULL; } } else if(images.type == NOL_ONE) { fprintf(stderr, "only one of --image/--window-image is allowed\n"); exit(EXIT_FAILURE); } char *image_path; int index = strtol(optarg, &image_path, 0); if(*image_path != ':') { fprintf(stderr, "--window-image should be in form ':'\n"); exit(EXIT_FAILURE); } else { image_path++; } list_arbitrary_insert(images.a, index, image_path); break; case 's': scaling_mode_str = optarg; break; case 't': scaling_mode_str = "tile"; break; case 'v': #if defined SWAY_GIT_VERSION && defined SWAY_GIT_BRANCH && defined SWAY_VERSION_DATE fprintf(stdout, "swaylock version %s (%s, branch \"%s\")\n", SWAY_GIT_VERSION, SWAY_VERSION_DATE, SWAY_GIT_BRANCH); #else fprintf(stdout, "version not detected\n"); #endif exit(EXIT_SUCCESS); break; default: fprintf(stderr, "%s", usage); exit(EXIT_FAILURE); } } enum scaling_mode scaling_mode = SCALING_MODE_STRETCH; if (strcmp(scaling_mode_str, "stretch") == 0) { scaling_mode = SCALING_MODE_STRETCH; } else if (strcmp(scaling_mode_str, "fill") == 0) { scaling_mode = SCALING_MODE_FILL; } else if (strcmp(scaling_mode_str, "fit") == 0) { scaling_mode = SCALING_MODE_FIT; } else if (strcmp(scaling_mode_str, "center") == 0) { scaling_mode = SCALING_MODE_CENTER; } else if (strcmp(scaling_mode_str, "tile") == 0) { scaling_mode = SCALING_MODE_TILE; } else { sway_abort("Unsupported scaling mode: %s", scaling_mode_str); } password_size = 1024; password = malloc(password_size); password[0] = '\0'; surfaces = create_list(); registry = registry_poll(); if (!registry->swaylock) { sway_abort("swaylock requires the compositor to support the swaylock extension."); } if (registry->pointer) { // We don't want swaylock to have a pointer wl_pointer_destroy(registry->pointer); registry->pointer = NULL; } for (i = 0; i < registry->outputs->length; ++i) { struct output_state *output = registry->outputs->items[i]; struct window *window = window_setup(registry, output->width, output->height, true); if (!window) { sway_abort("Failed to create surfaces."); } list_add(surfaces, window); } registry->input->notify = notify_key; if(images.type == NOL_ONE) { images.b = load_image(images.a); } else if(images.type == NOL_LIST) { for(i = 0; i <= ((list_t*) images.a)->length; ++i) { if(((list_t*) images.a)->items[i] != NULL) { list_arbitrary_insert(images.b, i, load_image(((list_t*) images.a)->items[i])); } } } for (i = 0; i < surfaces->length; ++i) { struct window *window = surfaces->items[i]; if (!window_prerender(window) || !window->cairo) { continue; } // always render the color in case the image doesn't fully cover the screen render_color(window, color); if (images.type == NOL_ONE) { render_image(window, images.b, scaling_mode); } else if(images.type == NOL_LIST) { if(((list_t*) images.b)->items[i] != NULL) { render_image(window, ((list_t*) images.b)->items[i], scaling_mode); } } window_render(window); } if(images.type == NOL_ONE) { cairo_surface_destroy(images.b); } else if(images.type == NOL_LIST) { for(i = 0; i <= ((list_t*) images.a)->length; ++i) { cairo_surface_destroy(((list_t*) images.b)->items[i]); } list_free(images.a); list_free(images.b); } bool locked = false; while (wl_display_dispatch(registry->display) != -1) { if (!locked) { for (i = 0; i < registry->outputs->length; ++i) { struct output_state *output = registry->outputs->items[i]; struct window *window = surfaces->items[i]; lock_set_lock_surface(registry->swaylock, output->output, window->surface); } locked = true; } } for (i = 0; i < surfaces->length; ++i) { struct window *window = surfaces->items[i]; window_teardown(window); } list_free(surfaces); registry_teardown(registry); return 0; }