cwm/calmwm.c

262 lines
5.8 KiB
C
Raw Permalink Normal View History

2007-04-27 21:58:48 +04:00
/*
* calmwm - the calm window manager
*
* Copyright (c) 2004 Marius Aamodt Eriksen <marius@monkey.org>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
2007-04-27 21:58:48 +04:00
*
* $Id$
*/
#include <sys/param.h>
#include <sys/queue.h>
#include <sys/wait.h>
#include <err.h>
#include <errno.h>
#include <getopt.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <unistd.h>
2007-04-27 21:58:48 +04:00
#include "calmwm.h"
Display *X_Dpy;
2007-04-27 21:58:48 +04:00
Cursor Cursor_move;
Cursor Cursor_resize;
Cursor Cursor_select;
Cursor Cursor_default;
Cursor Cursor_question;
2007-04-27 21:58:48 +04:00
struct screen_ctx_q Screenq = TAILQ_HEAD_INITIALIZER(Screenq);
struct client_ctx_q Clientq = TAILQ_HEAD_INITIALIZER(Clientq);
2007-04-27 21:58:48 +04:00
int HasXinerama, HasRandr, Randr_ev;
int Starting;
struct conf Conf;
2007-04-27 21:58:48 +04:00
static void sigchld_cb(int);
static void dpy_init(const char *);
static int x_errorhandler(Display *, XErrorEvent *);
static void x_setup(void);
static void x_setupscreen(struct screen_ctx *, u_int);
static void x_teardown(void);
2007-04-27 21:58:48 +04:00
int
main(int argc, char **argv)
{
const char *conf_file = NULL;
char *display_name = NULL;
int ch;
while ((ch = getopt(argc, argv, "c:d:")) != -1) {
2007-04-27 21:58:48 +04:00
switch (ch) {
case 'c':
2008-05-19 22:53:09 +04:00
conf_file = optarg;
break;
case 'd':
display_name = optarg;
break;
2007-04-27 21:58:48 +04:00
default:
usage();
2007-04-27 21:58:48 +04:00
}
}
argc -= optind;
2007-05-21 11:53:11 +04:00
argv += optind;
2007-04-27 21:58:48 +04:00
if (signal(SIGCHLD, sigchld_cb) == SIG_ERR)
2008-04-16 00:24:41 +04:00
err(1, "signal");
2007-04-27 21:58:48 +04:00
Starting = 1;
dpy_init(display_name);
2009-06-24 01:52:38 +04:00
bzero(&Conf, sizeof(Conf));
2008-05-19 22:53:09 +04:00
conf_setup(&Conf, conf_file);
xu_getatoms();
x_setup();
Starting = 0;
2007-04-27 21:58:48 +04:00
xev_loop();
x_teardown();
2007-04-27 21:58:48 +04:00
return (0);
}
static void
dpy_init(const char *dpyname)
2007-04-27 21:58:48 +04:00
{
int i;
2007-04-27 21:58:48 +04:00
if ((X_Dpy = XOpenDisplay(dpyname)) == NULL)
errx(1, "unable to open display \"%s\"",
XDisplayName(dpyname));
2007-04-27 21:58:48 +04:00
XSetErrorHandler(x_errorhandler);
HasRandr = XRRQueryExtension(X_Dpy, &Randr_ev, &i);
}
static void
x_setup(void)
{
struct screen_ctx *sc;
struct keybinding *kb;
int i;
for (i = 0; i < ScreenCount(X_Dpy); i++) {
sc = xcalloc(1, sizeof(*sc));
2007-04-27 21:58:48 +04:00
x_setupscreen(sc, i);
TAILQ_INSERT_TAIL(&Screenq, sc, entry);
2007-04-27 21:58:48 +04:00
}
/*
* XXX key grabs weren't done before, since Screenq was empty,
* do them here for now (this needs changing).
*/
TAILQ_FOREACH(kb, &Conf.keybindingq, entry)
conf_grab(&Conf, kb);
Cursor_move = XCreateFontCursor(X_Dpy, XC_fleur);
Cursor_resize = XCreateFontCursor(X_Dpy, XC_bottom_right_corner);
Cursor_select = XCreateFontCursor(X_Dpy, XC_hand1);
Cursor_default = XCreateFontCursor(X_Dpy, XC_X_cursor);
Cursor_question = XCreateFontCursor(X_Dpy, XC_question_arrow);
2007-04-27 21:58:48 +04:00
}
static void
x_teardown(void)
{
struct screen_ctx *sc;
TAILQ_FOREACH(sc, &Screenq, entry)
XFreeGC(X_Dpy, sc->gc);
XCloseDisplay(X_Dpy);
}
static void
2007-04-27 21:58:48 +04:00
x_setupscreen(struct screen_ctx *sc, u_int which)
{
Window *wins, w0, w1;
XWindowAttributes winattr;
XSetWindowAttributes rootattr;
int fake;
u_int nwins, i;
2007-04-27 21:58:48 +04:00
sc->which = which;
sc->rootwin = RootWindow(X_Dpy, sc->which);
conf_gap(&Conf, sc);
screen_update_geometry(sc, DisplayWidth(X_Dpy, sc->which),
DisplayHeight(X_Dpy, sc->which));
conf_color(&Conf, sc);
2007-04-27 21:58:48 +04:00
group_init(sc);
2007-04-27 21:58:48 +04:00
font_init(sc);
conf_font(&Conf, sc);
2007-04-27 21:58:48 +04:00
TAILQ_INIT(&sc->mruq);
/* Initialize menu window. */
menu_init(sc);
2007-04-27 21:58:48 +04:00
xu_setwmname(sc);
rootattr.event_mask = ChildMask|PropertyChangeMask|EnterWindowMask|
LeaveWindowMask|ColormapChangeMask|ButtonMask;
XChangeWindowAttributes(X_Dpy, sc->rootwin,
CWEventMask, &rootattr);
2007-04-27 21:58:48 +04:00
/* Deal with existing clients. */
2008-04-16 00:24:41 +04:00
XQueryTree(X_Dpy, sc->rootwin, &w0, &w1, &wins, &nwins);
2007-04-27 21:58:48 +04:00
for (i = 0; i < nwins; i++) {
XGetWindowAttributes(X_Dpy, wins[i], &winattr);
2007-04-27 21:58:48 +04:00
if (winattr.override_redirect ||
winattr.map_state != IsViewable)
2007-04-27 21:58:48 +04:00
continue;
client_new(wins[i], sc, winattr.map_state != IsUnmapped);
}
XFree(wins);
screen_updatestackingorder(sc);
2007-04-27 21:58:48 +04:00
if (XineramaQueryExtension(X_Dpy, &fake, &fake) == 1 &&
((HasXinerama = XineramaIsActive(X_Dpy)) == 1))
HasXinerama = 1;
if (HasRandr)
XRRSelectInput(X_Dpy, sc->rootwin, RRScreenChangeNotifyMask);
/*
* initial setup of xinerama screens, if we're using RandR then we'll
* redo this whenever the screen changes since a CTRC may have been
* added or removed
*/
screen_init_xinerama(sc);
XSync(X_Dpy, False);
2007-04-27 21:58:48 +04:00
}
static int
2007-04-27 21:58:48 +04:00
x_errorhandler(Display *dpy, XErrorEvent *e)
{
#ifdef DEBUG
{
char msg[80], number[80], req[80];
XGetErrorText(X_Dpy, e->error_code, msg, sizeof(msg));
2007-04-27 21:58:48 +04:00
snprintf(number, sizeof(number), "%d", e->request_code);
XGetErrorDatabaseText(X_Dpy, "XRequest", number,
2007-04-27 21:58:48 +04:00
"<unknown>", req, sizeof(req));
warnx("%s(0x%x): %s", req, (u_int)e->resourceid, msg);
}
#endif
2008-04-16 00:24:41 +04:00
if (Starting &&
e->error_code == BadAccess &&
e->request_code == X_GrabKey)
2007-04-27 21:58:48 +04:00
errx(1, "root window unavailable - perhaps another "
2008-04-16 00:24:41 +04:00
"wm is running?");
2007-04-27 21:58:48 +04:00
return (0);
}
static void
sigchld_cb(int which)
2007-04-27 21:58:48 +04:00
{
pid_t pid;
int save_errno = errno;
int status;
2007-04-27 21:58:48 +04:00
2008-04-16 00:24:41 +04:00
/* Collect dead children. */
while ((pid = waitpid(-1, &status, WNOHANG)) > 0 ||
(pid < 0 && errno == EINTR))
2007-04-27 21:58:48 +04:00
;
errno = save_errno;
2007-04-27 21:58:48 +04:00
}
__dead void
usage(void)
{
extern char *__progname;
fprintf(stderr, "usage: %s [-c file] [-d display]\n", __progname);
exit(1);
}