2013-04-24 21:05:53 +02:00
|
|
|
/* This is fcwd written by Adrien Schildknecht (c) 2013
|
|
|
|
* Email: adrien+dev@schischi.me
|
|
|
|
* Feel free to copy and redistribute in terms of the
|
|
|
|
* BSD license
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2013-04-26 01:11:46 +02:00
|
|
|
#include <strings.h>
|
2013-04-24 21:05:53 +02:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <glob.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
|
2013-04-26 00:50:37 +02:00
|
|
|
#define DEBUG 0
|
2013-04-26 00:40:51 +02:00
|
|
|
|
2013-04-25 20:27:50 +02:00
|
|
|
#define XA_STRING (XInternAtom(dpy, "STRING", 0))
|
|
|
|
#define XA_CARDINAL (XInternAtom(dpy, "CARDINAL", 0))
|
|
|
|
#define XA_WM_STATE (XInternAtom(dpy, "WM_STATE", 0))
|
2013-04-24 21:05:53 +02:00
|
|
|
|
2013-04-26 00:40:51 +02:00
|
|
|
#define LOG(fmt, ...) \
|
|
|
|
do { if (DEBUG) fprintf(stderr, fmt, __VA_ARGS__); } while (0)
|
|
|
|
|
2013-04-24 21:05:53 +02:00
|
|
|
Display *dpy;
|
|
|
|
|
|
|
|
typedef struct processes_s *processes_t;
|
|
|
|
struct processes_s {
|
|
|
|
struct proc_s {
|
|
|
|
long pid;
|
|
|
|
long ppid;
|
|
|
|
char name[32];
|
|
|
|
} *ps;
|
|
|
|
size_t n;
|
|
|
|
};
|
|
|
|
|
|
|
|
int nameCmp(const void *p1, const void *p2)
|
|
|
|
{
|
|
|
|
return strcasecmp(((struct proc_s *)p1)->name,
|
|
|
|
((struct proc_s *)p2)->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ppidCmp(const void *p1, const void *p2)
|
|
|
|
{
|
|
|
|
return ((struct proc_s *)p1)->ppid - ((struct proc_s *)p2)->ppid;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Window focusedWindow()
|
|
|
|
{
|
2013-04-26 00:40:51 +02:00
|
|
|
Atom type;
|
|
|
|
Window focuswin, root, *children;
|
|
|
|
int format, status, focusrevert;
|
2013-04-25 20:27:50 +02:00
|
|
|
unsigned long nitems, after;
|
|
|
|
unsigned char *data;
|
|
|
|
unsigned int nchildren;
|
2013-04-24 21:05:53 +02:00
|
|
|
|
|
|
|
dpy = XOpenDisplay (NULL);
|
|
|
|
if (!dpy)
|
|
|
|
exit (1);
|
|
|
|
XGetInputFocus (dpy, &focuswin, &focusrevert);
|
2013-04-25 20:27:50 +02:00
|
|
|
root = XDefaultRootWindow(dpy);
|
|
|
|
|
|
|
|
do {
|
2013-04-26 00:40:51 +02:00
|
|
|
status = XGetWindowProperty(dpy, focuswin, XA_WM_STATE, 0, 1024, 0,
|
2013-04-25 20:27:50 +02:00
|
|
|
XA_WM_STATE, &type, &format, &nitems, &after, &data);
|
2013-05-01 20:09:51 +02:00
|
|
|
if (status == Success) {
|
|
|
|
if (data) {
|
|
|
|
XFree(data);
|
|
|
|
LOG("Window ID = %lu\n", focuswin);
|
|
|
|
return focuswin;
|
|
|
|
}
|
2013-04-25 20:27:50 +02:00
|
|
|
}
|
2013-04-26 00:40:51 +02:00
|
|
|
else
|
|
|
|
return 0;
|
2013-04-25 20:27:50 +02:00
|
|
|
XQueryTree(dpy, focuswin, &root, &focuswin, &children, &nchildren);
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("%s", "Current window does not have WM_STATE, getting parent\n");
|
2013-04-25 20:27:50 +02:00
|
|
|
} while(focuswin != root);
|
|
|
|
|
|
|
|
return 0;
|
2013-04-24 21:05:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static long windowPid(Window focuswin)
|
|
|
|
{
|
|
|
|
Atom nameAtom = XInternAtom(dpy, "_NET_WM_PID", 1);
|
|
|
|
Atom type;
|
2013-04-25 20:27:50 +02:00
|
|
|
int format, status;
|
2013-04-24 21:05:53 +02:00
|
|
|
long pid = -1;
|
|
|
|
unsigned long nitems, after;
|
2013-04-25 20:27:50 +02:00
|
|
|
unsigned char *data;
|
2013-04-24 21:05:53 +02:00
|
|
|
|
2013-04-26 00:40:51 +02:00
|
|
|
status = XGetWindowProperty(dpy, focuswin, nameAtom, 0, 1024, 0,
|
2013-04-25 20:27:50 +02:00
|
|
|
XA_CARDINAL, &type, &format, &nitems, &after, &data);
|
2013-04-26 00:40:51 +02:00
|
|
|
if (status == Success && data) {
|
|
|
|
pid = *((long*)data);
|
|
|
|
XFree(data);
|
|
|
|
LOG("_NET_WM_PID = %lu\n", pid);
|
2013-04-24 21:05:53 +02:00
|
|
|
}
|
|
|
|
else
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("%s", "_NET_WM_PID not found\n");
|
2013-04-24 21:05:53 +02:00
|
|
|
return pid;
|
|
|
|
}
|
|
|
|
|
2013-05-17 21:53:49 +02:00
|
|
|
static char* windowStrings(Window focuswin, long unsigned int *size, char* hint)
|
2013-04-24 21:05:53 +02:00
|
|
|
{
|
|
|
|
Atom nameAtom = XInternAtom(dpy, hint, 1);
|
|
|
|
Atom type;
|
|
|
|
int format;
|
2013-04-26 01:11:46 +02:00
|
|
|
unsigned int i;
|
2013-04-24 21:05:53 +02:00
|
|
|
unsigned long after;
|
|
|
|
unsigned char *data = 0;
|
|
|
|
char *ret = NULL;
|
|
|
|
|
|
|
|
if (XGetWindowProperty(dpy, focuswin, nameAtom, 0, 1024, 0, AnyPropertyType,
|
|
|
|
&type, &format, size, &after, &data) == Success) {
|
|
|
|
if (data) {
|
2013-04-26 00:40:51 +02:00
|
|
|
if (type == XA_STRING) {
|
2013-04-24 21:05:53 +02:00
|
|
|
ret = malloc(sizeof(char) * *size);
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("%s = ", hint);
|
|
|
|
for (i = 0; i < *size; ++i) {
|
|
|
|
LOG("%c", data[i] == 0 ? ' ' : data[i]);
|
2013-04-24 21:05:53 +02:00
|
|
|
ret[i] = data[i];
|
|
|
|
}
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("%s", "\n");
|
2013-04-24 21:05:53 +02:00
|
|
|
}
|
|
|
|
XFree(data);
|
|
|
|
}
|
|
|
|
else
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("%s not found\n", hint);
|
2013-04-24 21:05:53 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void freeProcesses(processes_t p)
|
|
|
|
{
|
|
|
|
free(p->ps);
|
|
|
|
free(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
static processes_t getProcesses(void)
|
|
|
|
{
|
|
|
|
glob_t globbuf;
|
|
|
|
unsigned int i, j;
|
|
|
|
processes_t p;
|
|
|
|
|
|
|
|
glob("/proc/[0-9]*", GLOB_NOSORT, NULL, &globbuf);
|
|
|
|
p = malloc(sizeof(struct processes_s));
|
|
|
|
p->ps = malloc(globbuf.gl_pathc * sizeof(struct proc_s));
|
|
|
|
|
2013-05-17 21:53:49 +02:00
|
|
|
LOG("Found %zu processes\n", globbuf.gl_pathc);
|
2013-04-24 21:05:53 +02:00
|
|
|
for (i = j = 0; i < globbuf.gl_pathc; i++) {
|
|
|
|
char name[32];
|
|
|
|
FILE *tn;
|
|
|
|
|
|
|
|
(void)globbuf.gl_pathv[globbuf.gl_pathc - i - 1];
|
|
|
|
snprintf(name, sizeof(name), "%s%s",
|
|
|
|
globbuf.gl_pathv[globbuf.gl_pathc - i - 1], "/stat");
|
|
|
|
tn = fopen(name, "r");
|
|
|
|
if (tn == NULL)
|
|
|
|
continue;
|
2013-05-21 15:54:12 +02:00
|
|
|
if(fscanf(tn, "%ld (%32[^)] %*3c %ld", &p->ps[j].pid,
|
|
|
|
p->ps[j].name, &p->ps[j].ppid) != 3)
|
|
|
|
return NULL;
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("\t%-20s\tpid=%6ld\tppid=%6ld\n", p->ps[j].name, p->ps[j].pid,
|
|
|
|
p->ps[j].ppid);
|
2013-04-24 21:05:53 +02:00
|
|
|
fclose(tn);
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
p->n = j;
|
|
|
|
globfree(&globbuf);
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2013-05-01 20:09:51 +02:00
|
|
|
static int readPath(long pid)
|
2013-04-24 21:05:53 +02:00
|
|
|
{
|
|
|
|
char buf[255];
|
|
|
|
char path[64];
|
2013-04-26 00:40:51 +02:00
|
|
|
ssize_t len;
|
|
|
|
|
2013-04-24 21:05:53 +02:00
|
|
|
snprintf(path, sizeof(path), "/proc/%ld/cwd", pid);
|
2013-04-26 00:40:51 +02:00
|
|
|
if ((len = readlink(path, buf, 255)) != -1)
|
|
|
|
buf[len] = '\0';
|
2013-05-01 20:09:51 +02:00
|
|
|
if(len <= 0) {
|
|
|
|
LOG("Error readlink %s\n", path);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
LOG("Read %s\n", path);
|
2013-04-24 21:05:53 +02:00
|
|
|
fprintf(stdout, "%s\n", buf);
|
2013-05-01 20:09:51 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cwdOfDeepestChild(processes_t p, long pid)
|
|
|
|
{
|
|
|
|
int i;
|
2013-05-01 21:53:00 +02:00
|
|
|
struct proc_s key = {.ppid = pid}, *res = NULL, *lastRes = NULL;
|
2013-05-01 20:09:51 +02:00
|
|
|
|
|
|
|
do {
|
2013-05-01 21:53:00 +02:00
|
|
|
if(res) {
|
|
|
|
lastRes = res;
|
|
|
|
key.ppid = res->pid;
|
|
|
|
}
|
2013-05-01 20:09:51 +02:00
|
|
|
res = (struct proc_s *)bsearch(&key, p->ps, p->n,
|
|
|
|
sizeof(struct proc_s), ppidCmp);
|
2013-05-01 21:53:00 +02:00
|
|
|
} while(res);
|
|
|
|
|
|
|
|
if(!lastRes) {
|
|
|
|
readPath(pid);
|
|
|
|
return;
|
|
|
|
}
|
2013-05-01 20:09:51 +02:00
|
|
|
|
|
|
|
for(i = 0; lastRes != p->ps && (lastRes - i)->ppid == lastRes->ppid; ++i)
|
|
|
|
if(readPath((lastRes - i)->pid))
|
|
|
|
return;
|
|
|
|
for(i = 1; lastRes != p->ps + p->n && (lastRes + i)->ppid == lastRes->ppid; ++i)
|
|
|
|
if(readPath((lastRes + i)->pid))
|
|
|
|
return;
|
2013-04-24 21:05:53 +02:00
|
|
|
}
|
|
|
|
|
2013-05-21 15:54:12 +02:00
|
|
|
int getHomeDirectory()
|
|
|
|
{
|
|
|
|
LOG("%s", "getenv $HOME...\n");
|
|
|
|
fprintf(stdout, "%s\n", getenv("HOME"));
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2013-04-24 21:05:53 +02:00
|
|
|
int main(int argc, const char *argv[])
|
|
|
|
{
|
2013-04-26 01:11:46 +02:00
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
2013-04-24 21:05:53 +02:00
|
|
|
processes_t p;
|
|
|
|
long pid;
|
2013-04-25 20:27:50 +02:00
|
|
|
Window w = focusedWindow();
|
2013-05-21 15:54:12 +02:00
|
|
|
if (w == 0)
|
|
|
|
return getHomeDirectory();
|
2013-04-24 21:05:53 +02:00
|
|
|
|
|
|
|
pid = windowPid(w);
|
|
|
|
p = getProcesses();
|
2013-05-21 15:54:12 +02:00
|
|
|
if(p == NULL)
|
|
|
|
return getHomeDirectory();
|
2013-04-26 00:40:51 +02:00
|
|
|
if (pid != -1) {
|
2013-04-24 21:05:53 +02:00
|
|
|
qsort(p->ps, p->n, sizeof(struct proc_s), ppidCmp);
|
|
|
|
}
|
|
|
|
else {
|
2013-05-17 21:53:49 +02:00
|
|
|
long unsigned int size;
|
2013-04-26 01:11:46 +02:00
|
|
|
unsigned int i;
|
2013-05-17 21:53:49 +02:00
|
|
|
char* strings;
|
2013-04-24 21:05:53 +02:00
|
|
|
struct proc_s *res = NULL, key;
|
2013-04-26 01:11:46 +02:00
|
|
|
|
2013-04-24 21:05:53 +02:00
|
|
|
qsort(p->ps, p->n, sizeof(struct proc_s), nameCmp);
|
|
|
|
strings = windowStrings(w, &size, "WM_CLASS");
|
|
|
|
for(i = 0; i < size; i += strlen(strings + i) + 1) {
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("pidof %s\n", strings + i);
|
2013-04-24 21:05:53 +02:00
|
|
|
strcpy(key.name, strings + i);
|
|
|
|
res = (struct proc_s *)bsearch(&key, p->ps, p->n,
|
2013-04-26 00:40:51 +02:00
|
|
|
sizeof(struct proc_s), nameCmp);
|
2013-04-24 21:05:53 +02:00
|
|
|
if(res)
|
|
|
|
break;
|
|
|
|
}
|
2013-04-26 00:40:51 +02:00
|
|
|
if (res) {
|
2013-04-24 21:05:53 +02:00
|
|
|
pid = res->pid;
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("Found %s (%ld)\n", res->name, res->pid);
|
2013-04-24 21:05:53 +02:00
|
|
|
}
|
2013-04-26 00:40:51 +02:00
|
|
|
if (size != 0)
|
2013-04-24 21:05:53 +02:00
|
|
|
free(strings);
|
|
|
|
}
|
2013-05-01 20:09:51 +02:00
|
|
|
if (pid != -1)
|
|
|
|
cwdOfDeepestChild(p, pid);
|
2013-04-24 21:05:53 +02:00
|
|
|
else {
|
2013-04-26 00:40:51 +02:00
|
|
|
LOG("%s", "getenv $HOME...\n");
|
2013-04-24 21:05:53 +02:00
|
|
|
fprintf(stdout, "%s\n", getenv("HOME"));
|
|
|
|
}
|
|
|
|
freeProcesses(p);
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|