aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--README12
-rw-r--r--src/haiku.c2
-rw-r--r--src/settings.c3
-rw-r--r--src/state.c141
-rw-r--r--src/state.h9
5 files changed, 161 insertions, 6 deletions
diff --git a/README b/README
index 4260b90..be663a0 100644
--- a/README
+++ b/README
@@ -127,11 +127,15 @@ SIGNALS
order they where selected.
ENVIRONMENT
+ XDG_RUNTIME_DIR
+ This environment variable names the directory in which
+ the state file is stored. If unset or empty, /run is
+ used.
+
RADHARC_STATE
- The pathname to the state file, will be
- determined automatically if not set.
- If not set, you may only have one instance
- running per display server instance.
+ The pathname to the state file, will be determined
+ automatically if not set. If not set, you may only
+ have one instance running per display server instance.
RATIONALE
Your location is determined use GeoClue because it clunky,
diff --git a/src/haiku.c b/src/haiku.c
index d38fc9a..5c9e5c7 100644
--- a/src/haiku.c
+++ b/src/haiku.c
@@ -73,7 +73,7 @@ random_haiku(const char *str, ... /*, NULL */)
*/
void haiku(const char *s)
{
-#define HAIKU(...) do { fprintf(stderr, "\n%s", random_haiku(__VA_ARGS__, NULL)); return; } while (0)
+#define HAIKU(...) do { fprintf(stderr, "%s", random_haiku(__VA_ARGS__, NULL)); return; } while (0)
/* Yeah, I now most of these are in 5–7–5 syllables,
* rather than 5–7–5 mora. But really, how cares. */
diff --git a/src/settings.c b/src/settings.c
index fc527e7..6ee5130 100644
--- a/src/settings.c
+++ b/src/settings.c
@@ -229,12 +229,13 @@ parse_command_line(int argc, char *argv[], struct settings *settings)
if (!location_set && !(settings->temp)) {
fprintf(stderr,
- "%s: The -l option is mandatory, unless single value -t is used."
+ "%s: The -l option is mandatory, unless single value -t is used. "
"See `man 1 radharc` for more information.\n",
!argv0 ? "radharc" : argv0);
exit(2);
}
+ return;
fail:
haiku(argv0 ? argv0 : "radharc");
exit(1);
diff --git a/src/state.c b/src/state.c
index bad38eb..20427b8 100644
--- a/src/state.c
+++ b/src/state.c
@@ -16,6 +16,14 @@
*/
#include "state.h"
#include "solar.h"
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <errno.h>
+
+#include <libgamma.h>
+
+#define t(...) do { if (!(__VA_ARGS__)) goto fail; } while (0)
@@ -35,3 +43,136 @@ get_darkness(double elevation)
return NIGHT;
}
+
+/**
+ * Compare two display server environment strings.
+ *
+ * @param a_ One of the string.
+ * @param b_ The other string.
+ * @return -1, 0, or +1.
+ */
+static int
+displayenvcmp(const void *a_, const void *b_)
+{
+#define S(V, CD) ((V = ((CD)[1] == 'r' ? strrchr : strchr)(V, (CD)[0])))
+#ifdef __GNUC__
+# pragma GCC diagnostic push
+# pragma GCC diagnostic ignored "-Wdiscarded-qualifiers"
+#endif
+ char *a = a_;
+ char *b = b_;
+#ifdef __GNUC__
+# pragma GCC diagnostic pop
+#endif
+ char *p = strchr(a, '=') + 1;
+ char *q = strchr(b, '=') + 1;
+ int rc;
+
+ if ((*p != '/') && S(p, ":r") && S(p, ".l")) *p = '\0'; else p = NULL;
+ if ((*q != '/') && S(p, ":r") && S(p, ".l")) *q = '\0'; else q = NULL;
+
+ rc = strcmp(a, b);
+
+ if (*p) *p = '.';
+ if (*q) *q = '.';
+ return rc;
+#undef S
+}
+
+
+/**
+ * Make an display string safe for a pathname.
+ *
+ * @param str The string.
+ * @return A safe version of the string, `NULL` on error.
+ */
+static char *
+escape_display(const char* str)
+{
+ char *r, *w, *rc = malloc((2 * strlen(str) + 1) * sizeof(char *));
+ int s = 0;
+ if (!rc) return NULL;
+ for (r = w = strchr(rc, '=') + 1; *r; r++) {
+ if (!s || (*r != '/')) {
+ if (strchr("@=/", *r)) *w++ = '@';
+ *w++ = (*r == '/' ? 's' : *r);
+ s = (*r == '/');
+ }
+ }
+ if (s) w[-2] = '\0';
+ return rc;
+}
+
+
+/**
+ * The string of display servers.
+ *
+ * @param settings The settings.
+ * @return The string, `NULL` on error.
+ */
+static char *
+get_display_string(struct settings *settings)
+{
+ const char *var, *val;
+ char *r, *d = NULL, *rc = NULL, **displays;
+ size_t i, n = 0, len = 0;
+ int method, saved_errno;
+
+ t (displays = malloc(settings->monitors_n * sizeof(char *)));
+ for (i = 0; i < settings->monitors_n; i++)
+ if ((settings->monitors_arg[i] == 'd') && strchr(settings->monitors_id[i], '='))
+ len += 1 + strlen(displays[n++] = settings->monitors_id[i]);
+ if (n) goto custom;
+
+ if (!libgamma_list_methods(&method, 1, 0)) {
+ fprintf(stderr, "No display was found.\n"
+ "DRM support missing.\n"
+ "Can you even see?\n");
+ return errno = 0, NULL;
+ }
+
+ var = libgamma_method_default_site_variable(method);
+ val = libgamma_method_default_site(method);
+ if (!val) return strdup("");
+ t (d = malloc((3 + strlen(var) + strlen(val)) * sizeof(char)));
+ stpcpy(stpcpy(stpcpy(stpcpy(d, "."), var), "="), val);
+ t (rc = escape_display(d));
+ return rc;
+
+custom:
+ qsort(displays, n, sizeof(*displays), displayenvcmp);
+ t (r = rc = malloc((2 * len + 1) * sizeof(char)));
+ for (i = 0; i < n; i++) {
+ t (d = escape_display(displays[i]));
+ r = stpcpy(stpcpy(r, "."), d), free(d), d = NULL;
+ }
+ return rc;
+
+fail:
+ saved_errno = errno, free(rc), free(d), errno = saved_errno;
+ return NULL;
+}
+
+
+/**
+ * Set $RADHARC_STATE.
+ *
+ * @param settings The settings.
+ * @return 0 on success, -1 on error.
+ */
+int
+get_state_pathname(struct settings *settings)
+{
+ const char *dir = getenv("XGD_RUNTIME_DIR");
+ char *display;
+ char *env;
+ int rc = -1, saved_errno;
+ t (display = get_display_string(settings));
+ if (!dir || !*dir) dir = "/run";
+ t (env = malloc((strlen(dir) + sizeof("/radharc/") + strlen(display)) * sizeof(char)));
+ stpcpy(stpcpy(stpcpy(env, dir), "/radharc/"), display);
+ rc = setenv("RADHARC_STATE", env, 1);
+fail:
+ return saved_errno = errno, free(env), free(display), errno = saved_errno, rc;
+}
+
diff --git a/src/state.h b/src/state.h
index cd4cef6..bcb271b 100644
--- a/src/state.h
+++ b/src/state.h
@@ -14,6 +14,7 @@
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
+#include "settings.h"
@@ -63,3 +64,11 @@ enum darkness {
*/
enum darkness get_darkness(double elevation);
+/**
+ * Set $RADHARC_STATE.
+ *
+ * @param settings The settings.
+ * @return 0 on success, -1 on error.
+ */
+int get_state_pathname(struct settings *settings);
+