aboutsummaryrefslogtreecommitdiffstats
path: root/src/redshift.c
blob: d8969af5c1ca3025c8b590950a87a40c523ee835 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
/* redshift.c -- Main program source
 * This file is part of redshift-ng.
 * 
 * redshift-ng is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 * 
 * redshift-ng is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with redshift-ng.  If not, see <http://www.gnu.org/licenses/>.
 * 
 * Copyright (c) 2009-2017  Jon Lund Steffensen <jonlst@gmail.com>
 * Copyright (c) 2025       Mattias Andrée <m@maandree.se>
 */
#include "common.h"

/* poll.h is not available on Windows but there is no Windows location provider
 * using polling. On Windows, we just define some stubs to make things compile.
 */
#ifndef WINDOWS
# include <poll.h>
#else
#define POLLIN 0
struct pollfd {
	int fd;
	short events;
	short revents;
};
int poll(struct pollfd *fds, int nfds, int timeout) { abort(); }
#endif


/* Duration of sleep between screen updates (milliseconds). */
#define SLEEP_DURATION        5000
#define SLEEP_DURATION_SHORT  100

/* Length of fade in numbers of short sleep durations. */
#define FADE_LENGTH  40


const struct gamma_method *gamma_methods[] = {
#ifdef ENABLE_COOPGAMMA
	&coopgamma_gamma_method,
#endif
#ifdef ENABLE_DRM
	&drm_gamma_method,
#endif
#ifdef ENABLE_RANDR
	&randr_gamma_method,
#endif
#ifdef ENABLE_VIDMODE
	&vidmode_gamma_method,
#endif
#ifdef ENABLE_QUARTZ
	&quartz_gamma_method,
#endif
#ifdef ENABLE_WINGDI
	&w32gdi_gamma_method,
#endif
	&dummy_gamma_method,
	NULL
};


const struct location_provider *location_providers[] = {
#ifdef ENABLE_GEOCLUE2
	&geoclue2_location_provider,
#endif
#ifdef ENABLE_CORELOCATION
	&corelocation_location_provider,
#endif
	&manual_location_provider,
	NULL
};


/* Names of periods of day */
static const char *period_names[] = {
	/* TRANSLATORS: Name printed when period of day is unknown */
	N_("None"),
	N_("Daytime"),
	N_("Night"),
	N_("Transition")
};


#if defined(__GNUC__)
# pragma GCC diagnostic push
# pragma GCC diagnostic ignored "-Wfloat-equal"
#endif
static int
exact_eq(double a, double b)
{
	return a == b;
}
#if defined(__GNUC__)
# pragma GCC diagnostic pop
#endif


/* Determine which period we are currently in based on time offset. */
static enum period
get_period_from_time(const struct transition_scheme *transition, int time_offset)
{
	if (time_offset < transition->dawn.start || time_offset >= transition->dusk.end)
		return PERIOD_NIGHT;
	else if (time_offset >= transition->dawn.end && time_offset < transition->dusk.start)
		return PERIOD_DAYTIME;
	else
		return PERIOD_TRANSITION;
}

/* Determine which period we are currently in based on solar elevation. */
static enum period
get_period_from_elevation(const struct transition_scheme *transition, double elevation)
{
	if (elevation < transition->low)
		return PERIOD_NIGHT;
	else if (elevation < transition->high)
		return PERIOD_TRANSITION;
	else
		return PERIOD_DAYTIME;
}

/* Determine how far through the transition we are based on time offset. */
static double
get_transition_progress_from_time(const struct transition_scheme *transition, int time_offset)
{
	if (time_offset < transition->dawn.start || time_offset >= transition->dusk.end)
		return 0.0;
	else if (time_offset < transition->dawn.end)
		return (transition->dawn.start - time_offset) / (double)(transition->dawn.start - transition->dawn.end);
	else if (time_offset > transition->dusk.start)
		return (transition->dusk.end - time_offset) / (double)(transition->dusk.end - transition->dusk.start);
	else
		return 1.0;
}

/* Determine how far through the transition we are based on elevation. */
static double
get_transition_progress_from_elevation(const struct transition_scheme *transition, double elevation)
{
	if (elevation < transition->low)
		return 0.0;
	else if (elevation < transition->high)
		return (transition->low - elevation) / (transition->low - transition->high);
	else
		return 1.0;
}

/* Return number of seconds since midnight from timestamp. */
static int
get_seconds_since_midnight(double timestamp)
{
	time_t t = (time_t)timestamp;
	struct tm tm;
	localtime_r(&t, &tm);
	return tm.tm_sec + tm.tm_min * 60 + tm.tm_hour * 3600;
}

/* Print verbose description of the given period. */
static void
print_period(enum period period, double transition)
{
	if (period == PERIOD_TRANSITION)
		printf(_("Period: %s (%.2f%% day)\n"), gettext(period_names[period]), transition * 100);
	else
		printf(_("Period: %s\n"), gettext(period_names[period]));
}

/* Print location */
static void
print_location(const struct location *location)
{
	/* TRANSLATORS: Abbreviation for `north' */
	const char *north = _("N");
	/* TRANSLATORS: Abbreviation for `south' */
	const char *south = _("S");
	/* TRANSLATORS: Abbreviation for `east' */
	const char *east = _("E");
	/* TRANSLATORS: Abbreviation for `west' */
	const char *west = _("W");

	/* TRANSLATORS: Append degree symbols after %f if possible.
	   The string following each number is an abreviation for
	   north, source, east or west (N, S, E, W). */
	printf(_("Location: %.2f %s, %.2f %s\n"),
	       fabs(location->latitude), location->latitude >= 0.0 ? north : south,
	       fabs(location->longitude), location->longitude >= 0.0 ? east : west);
}

/* Interpolate colour setting structs given alpha. */
static void
interpolate_colour_settings(const struct colour_setting *first, const struct colour_setting *second,
                            double alpha, struct colour_setting *result)
{
	int i;
	alpha = CLAMP(0.0, alpha, 1.0);
	result->temperature = (1.0 - alpha) * first->temperature + alpha * second->temperature;
	result->brightness  = (1.0 - alpha) * first->brightness  + alpha * second->brightness;
	for (i = 0; i < 3; i++)
		result->gamma[i] = (1.0 - alpha) * first->gamma[i] + alpha * second->gamma[i];
}

/* Interpolate colour setting structs transition scheme. */
static void
interpolate_transition_scheme(const struct transition_scheme *transition, double alpha, struct colour_setting *result)
{
	interpolate_colour_settings(&transition->night, &transition->day, alpha, result);
}

/* Return 1 if colour settings have major differences, otherwise 0.
   Used to determine if a fade should be applied in continual mode. */
static int
colour_setting_diff_is_major(const struct colour_setting *first, const struct colour_setting *second)
{
	return MAX(first->temperature, second->temperature) - MIN(first->temperature, second->temperature) > 25UL ||
	       fabs(first->brightness - second->brightness) > 0.1 ||
	       fabs(first->gamma[0] - second->gamma[0]) > 0.1 ||
	       fabs(first->gamma[1] - second->gamma[1]) > 0.1 ||
	       fabs(first->gamma[2] - second->gamma[2]) > 0.1;
}


static int
provider_try_start(const struct location_provider *provider, LOCATION_STATE **state,
                   struct config_ini_state *config, char *args)
{
	const char *manual_keys[] = {"lat", "lon"};
	struct config_ini_section *section;
	struct config_ini_setting *setting;
	char *next_arg, *value;
	const char *key;
	int i;

	if (provider->init(state) < 0) {
		weprintf(_("Initialization of %s failed."), provider->name);
		return -1;
	}

	/* Set provider options from config file. */
	if ((section = config_ini_get_section(config, provider->name))) {
		for (setting = section->settings; setting; setting = setting->next) {
			if (provider->set_option(*state, setting->name, setting->value) < 0) {
				provider->free(*state);
				weprintf(_("Failed to set %s option."), provider->name);
				/* TRANSLATORS: `help' must not be translated. */
				weprintf(_("Try `-l %s:help' for more information."), provider->name);
				return -1;
			}
		}
	}

	/* Set provider options from command line. */
	for (i = 0; args; i++) {
		next_arg = strchr(args, ':');
		if (next_arg)
			*next_arg++ = '\0';

		key = args;
		value = strchr(args, '=');
		if (!value) {
			/* The options for the "manual" method can be set
			   without keys on the command line for convencience
			   and for backwards compatability. We add the proper
			   keys here before calling set_option(). */
			if (!strcmp(provider->name, "manual") && i < ELEMSOF(manual_keys)) {
				key = manual_keys[i];
				value = args;
			} else {
				weprintf(_("Failed to parse option `%s'."), args);
				return -1;
			}
		} else {
			*value++ = '\0';
		}

		if (provider->set_option(*state, key, value) < 0) {
			provider->free(*state);
			weprintf(_("Failed to set %s option."), provider->name);
			/* TRANSLATORS: `help' must not be translated. */
			weprintf(_("Try `-l %s:help' for more information."), provider->name);
			return -1;
		}

		args = next_arg;
	}

	/* Start provider. */
	if (provider->start(*state) < 0) {
		provider->free(*state);
		weprintf(_("Failed to start provider %s.\n"), provider->name);
		return -1;
	}

	return 0;
}

static int
method_try_start(const struct gamma_method *method, GAMMA_STATE **state,
		 enum program_mode mode, struct config_ini_state *config, char *args)
{
	struct config_ini_section *section;
	struct config_ini_setting *setting;
	char *next_arg, *value;
	const char *key;

	if (method->init(state) < 0) {
		weprintf(_("Initialization of %s failed."), method->name);
		return -1;
	}

	/* Set method options from config file. */
	if ((section = config_ini_get_section(config, method->name))) {
		for (setting = section->settings; setting; setting = setting->next) {
			if (method->set_option(*state, setting->name, setting->value) < 0) {
				method->free(*state);
				weprintf(_("Failed to set %s option."), method->name);
				/* TRANSLATORS: `help' must not be translated. */
				weprintf(_("Try `-m %s:help' for more information.\n"), method->name); /* TODO \n */
				return -1;
			}
		}
	}

	/* Set method options from command line. */
	while (args) {
		next_arg = strchr(args, ':');
		if (next_arg)
			*next_arg++ = '\0';

		key = args;
		value = strchr(args, '=');
		if (!value) {
			weprintf(_("Failed to parse option `%s'."), args);
			return -1;
		}
		*value++ = '\0';

		if (method->set_option(*state, key, value) < 0) {
			method->free(*state);
			weprintf(_("Failed to set %s option."), method->name);
			/* TRANSLATORS: `help' must not be translated. */
			weprintf(_("Try -m %s:help' for more information.\n"), method->name); /* TODO missing ` and \n */
			return -1;
		}

		args = next_arg;
	}

	/* Start method. */
	if (method->start(*state, mode) < 0) {
		method->free(*state);
		weprintf(_("Failed to start adjustment method %s.\n"), method->name); /* TODO \n */
		return -1;
	}

	return 0;
}


/* Check whether gamma is within allowed levels. */
static int
gamma_is_valid(const double gamma[3])
{
	return WITHIN(MIN_GAMMA, gamma[0], MAX_GAMMA) &&
	       WITHIN(MIN_GAMMA, gamma[1], MAX_GAMMA) &&
	       WITHIN(MIN_GAMMA, gamma[2], MAX_GAMMA);
}

/* Check whether location is valid.
   Prints error message on stderr and returns 0 if invalid, otherwise
   returns 1. */
static int
location_is_valid(const struct location *location)
{
	if (!WITHIN(MIN_LATITUDE, location->latitude, MAX_LATITUDE)) {
		/* TRANSLATORS: Append degree symbols if possible. */
		weprintf(_("Latitude must be between %.1f and %.1f.\n"), MIN_LATITUDE, MAX_LATITUDE); /* TODO \n */
		return 0;
	}
	if (!WITHIN(MIN_LONGITUDE, location->longitude, MAX_LONGITUDE)) {
		/* TRANSLATORS: Append degree symbols if possible. */
		weprintf(_("Longitude must be between %.1f and %.1f.\n"), MIN_LONGITUDE, MAX_LONGITUDE); /* TODO \n */
		return 0;
	}
	return 1;
}

/* Wait for location to become available from provider.
   Waits until timeout (milliseconds) has elapsed or forever if timeout
   is -1. Writes location to loc. Returns -1 on error,
   0 if timeout was reached, 1 if location became available. */
static int
provider_get_location(const struct location_provider *provider, LOCATION_STATE *state, int timeout, struct location *loc)
{
	int r, available, loc_fd;
	struct pollfd pollfds[1];
	double now, later;

	do {
		loc_fd = provider->get_fd(state);
		if (loc_fd >= 0) {
			/* Provider is dynamic. */
			/* TODO: This should use a monotonic time source. */
			now = systemtime_get_time();

			/* Poll on file descriptor until ready. */
			pollfds[0].fd = loc_fd;
			pollfds[0].events = POLLIN;
			r = poll(pollfds, 1, timeout);
			if (r < 0) {
				weprintf("poll {{.fd=<location provider>, .events=EPOLLIN}} 1 %i:", timeout);
				return -1;
			} else if (r == 0) {
				return 0;
			}

			later = systemtime_get_time();

			/* Adjust timeout by elapsed time */
			if (timeout >= 0) {
				timeout -= (later - now) * 1000;
				timeout = MAX(timeout, 0);
			}
		}

		if (provider->handle(state, loc, &available) < 0)
			return -1;
	} while (!available);

	return 1;
}

/* Easing function for fade.
   See https://github.com/mietek/ease-tween */
static double
ease_fade(double t)
{
	if (t <= 0) return 0;
	if (t >= 1) return 1;
	return 1.0042954579734844 * exp(
		-6.4041738958415664 * exp(-7.2908241330981340 * t));
}


/* Run continual mode loop
   This is the main loop of the continual mode which keeps track of the
   current time and continuously updates the screen to the appropriate
   colour temperature. */
static void
run_continual_mode(const struct location_provider *provider, LOCATION_STATE *location_state,
                   const struct transition_scheme *scheme, const struct gamma_method *method,
                   GAMMA_STATE *method_state, int use_fade, int preserve_gamma, int verbose)
{
	int done = 0;
	int prev_disabled = 1;
	int disabled = 0;
	int location_available = 1;
	struct colour_setting fade_start_interp;
	struct colour_setting prev_target_interp;
	struct colour_setting interp;
	struct location loc;
	int need_location;

	/* Short fade parameters */
	int fade_length = 0;
	int fade_time = 0;

	/* Save previous parameters so we can avoid printing status updates if
	   the values did not change. */
	enum period prev_period = PERIOD_NONE;

	signals_install_handlers();

	/* Previous target colour setting and current actual colour setting.
	   Actual colour setting takes into account the current colour fade. */
	prev_target_interp = COLOUR_SETTING_NEUTRAL;

	interp = COLOUR_SETTING_NEUTRAL;

	loc = (struct location){NAN, NAN};
	need_location = !scheme->use_time;
	if (need_location) {
		weprintf(_("Waiting for initial location to become available...\n")); /* TODO \n */

		/* Get initial location from provider */
		if (provider_get_location(provider, location_state, -1, &loc) < 0)
			eprintf(_("Unable to get location from provider."));

		if (!location_is_valid(&loc))
			eprintf(_("Invalid location returned from provider.\n")); /* TODO \n */

		print_location(&loc);
	}

	if (verbose) {
		printf(_("Color temperature: %luK\n"), interp.temperature);
		printf(_("Brightness: %.2f\n"), interp.brightness);
	}

	/* Continuously adjust colour temperature */
	for (;;) {
		double now;
		enum period period;
		double transition_prog;
		struct colour_setting target_interp;
		int delay, loc_fd;

		/* Check to see if disable signal was caught */
		if (disable && !done) {
			disabled = !disabled;
			disable = 0;
		}

		/* Check to see if exit signal was caught */
		if (exiting) {
			if (done)
				break; /* On second signal stop the ongoing fade. */
			done = 1;
			disabled = 1;
			exiting = 0;
		}

		/* Print status change */
		if (verbose && disabled != prev_disabled)
			printf(_("Status: %s\n"), disabled ? _("Disabled") : _("Enabled"));

		prev_disabled = disabled;

		/* Read timestamp */
		now = systemtime_get_time();

		if (scheme->use_time) {
			int time_offset = get_seconds_since_midnight(now);

			period = get_period_from_time(scheme, time_offset);
			transition_prog = get_transition_progress_from_time(scheme, time_offset);
		} else {
			/* Current angular elevation of the sun */
			double elevation;
			if (libred_solar_elevation(loc.latitude, loc.longitude, &elevation))
				eprintf("libred_solar_elevation:");

			period = get_period_from_elevation(scheme, elevation);
			transition_prog = get_transition_progress_from_elevation(scheme, elevation);
		}

		/* Use transition progress to get target colour temperature. */
		interpolate_transition_scheme(scheme, transition_prog, &target_interp);

		if (disabled) {
			period = PERIOD_NONE;
			target_interp = COLOUR_SETTING_NEUTRAL;
		}

		if (done)
			period = PERIOD_NONE;

		/* Print period if it changed during this update,
		   or if we are in the transition period. In transition we
		   print the progress, so we always print it in
		   that case. */
		if (verbose && (period != prev_period || period == PERIOD_TRANSITION))
			print_period(period, transition_prog);

		/* Activate hooks if period changed */
		if (period != prev_period)
			hooks_signal_period_change(prev_period, period);

		/* Start fade if the parameter differences are too big to apply
		   instantly. */
		if (use_fade) {
			if (fade_length ? colour_setting_diff_is_major(&target_interp, &prev_target_interp)
			                : colour_setting_diff_is_major(&interp, &target_interp)) {
				fade_length = FADE_LENGTH;
				fade_time = 0;
				fade_start_interp = interp;
			}
		}

		/* Handle ongoing fade */
		if (fade_length != 0) {
			double frac = ++fade_time / (double)fade_length;
			double alpha = CLAMP(0.0, ease_fade(frac), 1.0);

			interpolate_colour_settings(&fade_start_interp, &target_interp, alpha, &interp);

			if (fade_time > fade_length) {
				fade_time = 0;
				fade_length = 0;
			}
		} else {
			interp = target_interp;
		}

		/* Break loop when done and final fade is over */
		if (done && fade_length == 0)
			break;

		if (verbose) {
			if (prev_target_interp.temperature != target_interp.temperature)
				printf(_("Color temperature: %luK\n"), target_interp.temperature);
			if (!exact_eq(prev_target_interp.brightness, target_interp.brightness))
				printf(_("Brightness: %.2f\n"), target_interp.brightness);
		}

		/* Adjust temperature */
		if (method->set_temperature(method_state, &interp, preserve_gamma) < 0)
			eprintf(_("Temperature adjustment failed."));

		/* Save period and target colour setting as previous */
		prev_period = period;
		prev_target_interp = target_interp;

		/* Sleep length depends on whether a fade is ongoing. */
		delay = fade_length ? SLEEP_DURATION_SHORT : SLEEP_DURATION;

		/* Update location. */
		loc_fd = need_location ? provider->get_fd(location_state) : -1;

		if (loc_fd >= 0) {
			struct pollfd pollfds[1];
			struct location new_loc;
			int r, new_available;

			/* Provider is dynamic. */
			pollfds[0].fd = loc_fd;
			pollfds[0].events = POLLIN;
			r = poll(pollfds, 1, delay);
			if (r < 0) {
				if (errno == EINTR)
					continue;
				weprintf("poll:");
				eprintf(_("Unable to get location from provider."));
			} else if (!r) {
				continue;
			}

			/* Get new location and availability
			   information. */
			if (provider->handle(location_state, &new_loc, &new_available) < 0)
				eprintf(_("Unable to get location from provider."));

			if (!new_available && new_available != location_available) {
				weprintf(_("Location is temporarily unavailable; Using previous" /* TODO captial U efter ; and \n*/
				           " location until it becomes available...\n"));
			}

			if (new_available &&
			    (!exact_eq(new_loc.latitude, loc.latitude) ||
			     !exact_eq(new_loc.longitude, loc.longitude) ||
			     new_available != location_available)) {
				loc = new_loc;
				print_location(&loc);
			}

			location_available = new_available;

			if (!location_is_valid(&loc))
				eprintf(_("Invalid location returned from provider."));
		} else {
			systemtime_msleep(delay);
		}
	}

	/* Restore saved gamma ramps */
	method->restore(method_state);
}


int
main(int argc, char *argv[])
{
	struct options options;
	struct config_ini_state config_state;
	struct transition_scheme *scheme;
	GAMMA_STATE *method_state;
	LOCATION_STATE *location_state;
	int need_location;
	size_t i;
	struct location loc = { NAN, NAN };
	double now, transition_prog;
	enum period period;
	struct colour_setting colour;

	argv0 = argv[0];

#ifdef ENABLE_NLS
	setlocale(LC_CTYPE, "");
	setlocale(LC_MESSAGES, "");
	bindtextdomain(PACKAGE, LOCALEDIR);
	textdomain(PACKAGE);
#endif

	options_init(&options);
	options_parse_args(&options, argc, argv);

	/* Load settings from config file. */
	config_ini_init(&config_state, options.config_filepath);

	free(options.config_filepath);

	options_parse_config_file(&options, &config_state);

	options_set_defaults(&options);

	if (options.scheme.dawn.start >= 0 || options.scheme.dawn.end >= 0 ||
	    options.scheme.dusk.start >= 0 || options.scheme.dusk.end >= 0) {
		if (options.scheme.dawn.start < 0 || options.scheme.dawn.end < 0 ||
		    options.scheme.dusk.start < 0 || options.scheme.dusk.end < 0)
			eprintf(_("Partial time-configuration not supported!"));

		if (options.scheme.dawn.start > options.scheme.dawn.end ||
		    options.scheme.dawn.end > options.scheme.dusk.start ||
		    options.scheme.dusk.start > options.scheme.dusk.end)
			eprintf(_("Invalid dawn/dusk time configuration!"));

		options.scheme.use_time = 1;
	}

	/* Initialize location provider if needed. If provider is NULL
	   try all providers until one that works is found. */

	/* Location is not needed for reset mode and manual mode. */
	need_location = options.mode != PROGRAM_MODE_RESET &&
	                options.mode != PROGRAM_MODE_MANUAL &&
	                !options.scheme.use_time;
	if (need_location) {
		if (options.provider) {
			/* Use provider specified on command line. */
			if (provider_try_start(options.provider, &location_state, &config_state, options.provider_args) < 0)
				exit(1);
		} else {
			/* Try all providers, use the first that works. */
			for (i = 0; location_providers[i]; i++) {
				const struct location_provider *p = location_providers[i];
				weprintf(_("Trying location provider `%s'..."), p->name);
				if (provider_try_start(p, &location_state, &config_state, NULL) < 0) {
					weprintf(_("Trying next provider..."));
					continue;
				}

				/* Found provider that works. */
				printf(_("Using provider `%s'.\n"), p->name);
				options.provider = p;
				break;
			}

			/* Failure if no providers were successful at this point. */
			if (!options.provider)
				eprintf(_("No more location providers to try."));
		}

		/* Solar elevations */
		if (options.scheme.high < options.scheme.low) {
			eprintf(_("High transition elevation cannot be lower than the low transition elevation."));
		}

		if (options.verbose) {
			/* TRANSLATORS: Append degree symbols if possible. */
			printf(_("Solar elevations: day above %.1f, night below %.1f\n"),
			       options.scheme.high, options.scheme.low);
		}
	}

	if (options.mode != PROGRAM_MODE_RESET &&
	    options.mode != PROGRAM_MODE_MANUAL) {
		if (options.verbose) {
			printf(_("Temperatures: %luK at day, %luK at night\n"),
			       options.scheme.day.temperature, options.scheme.night.temperature);
		}

		/* Colour temperature */
		if (!WITHIN(MIN_TEMPERATURE, options.scheme.day.temperature, MAX_TEMPERATURE) ||
		    !WITHIN(MIN_TEMPERATURE, options.scheme.night.temperature, MAX_TEMPERATURE))
			eprintf(_("Temperature must be between %luK and %luK."), MIN_TEMPERATURE, MAX_TEMPERATURE);
	}

	if (options.mode == PROGRAM_MODE_MANUAL) {
		/* Check colour temperature to be set */
		if (!WITHIN(MIN_TEMPERATURE, options.temp_set, MAX_TEMPERATURE))
			eprintf(_("Temperature must be between %luK and %luK."), MIN_TEMPERATURE, MAX_TEMPERATURE);
	}

	/* Brightness */
	if (!WITHIN(MIN_BRIGHTNESS, options.scheme.day.brightness, MAX_BRIGHTNESS) ||
	    !WITHIN(MIN_BRIGHTNESS, options.scheme.night.brightness, MAX_BRIGHTNESS))
		eprintf(_("Brightness values must be between %.1f and %.1f."), MIN_BRIGHTNESS, MAX_BRIGHTNESS);

	if (options.verbose)
		printf(_("Brightness: %.2f:%.2f\n"), options.scheme.day.brightness, options.scheme.night.brightness);

	/* Gamma */
	if (!gamma_is_valid(options.scheme.day.gamma) ||
	    !gamma_is_valid(options.scheme.night.gamma))
		eprintf(_("Gamma value must be between %.1f and %.1f."), MIN_GAMMA, MAX_GAMMA);

	if (options.verbose) {
		/* TRANSLATORS: The string in parenthesis is either
		   Daytime or Night (translated). */
		printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"),
		       _("Daytime"), options.scheme.day.gamma[0],
		       options.scheme.day.gamma[1],
		       options.scheme.day.gamma[2]);
		printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"),
		       _("Night"), options.scheme.night.gamma[0],
		       options.scheme.night.gamma[1],
		       options.scheme.night.gamma[2]);
	}

	scheme = &options.scheme;

	/* Initialize gamma adjustment method. If method is NULL
	   try all methods until one that works is found. */

	/* Gamma adjustment not needed for print mode */
	if (options.mode != PROGRAM_MODE_PRINT) {
		if (options.method) {
			/* Use method specified on command line. */
			if (method_try_start(options.method, &method_state, options.mode, &config_state, options.method_args) < 0)
				exit(1);
		} else {
			/* Try all methods, use the first that works. */
			for (i = 0; gamma_methods[i]; i++) {
				const struct gamma_method *m = gamma_methods[i];
				if (!m->autostart)
					continue;

				if (method_try_start(m, &method_state, options.mode, &config_state, NULL) < 0) {
					weprintf(_("Trying next method..."));
					continue;
				}

				/* Found method that works. */
				printf(_("Using method `%s'.\n"), m->name);
				options.method = m;
				break;
			}

			/* Failure if no methods were successful at this point. */
			if (!options.method)
				eprintf(_("No more methods to try."));
		}
	}

	config_ini_free(&config_state);

	switch (options.mode) {
	case PROGRAM_MODE_ONE_SHOT:
	case PROGRAM_MODE_PRINT:
		if (need_location) {
			weprintf(_("Waiting for current location to become available..."));

			/* Wait for location provider. */
			if (provider_get_location(options.provider, location_state, -1, &loc) < 0)
				eprintf(_("Unable to get location from provider."));

			if (!location_is_valid(&loc))
				exit(1);

			print_location(&loc);
		}

		now = systemtime_get_time();

		if (options.scheme.use_time) {
			int time_offset = get_seconds_since_midnight(now);
			period = get_period_from_time(scheme, time_offset);
			transition_prog = get_transition_progress_from_time(scheme, time_offset);
		} else {
			/* Current angular elevation of the sun */
			double elevation;
			if (libred_solar_elevation(loc.latitude, loc.longitude, &elevation))
				eprintf("libred_solar_elevation:");
			if (options.verbose) {
				/* TRANSLATORS: Append degree symbol if possible. */
				printf(_("Solar elevation: %f\n"), elevation);
			}

			period = get_period_from_elevation(scheme, elevation);
			transition_prog = get_transition_progress_from_elevation(scheme, elevation);
		}

		/* Use transition progress to set colour temperature */
		interpolate_transition_scheme(scheme, transition_prog, &colour);

		if (options.verbose || options.mode == PROGRAM_MODE_PRINT) {
			print_period(period, transition_prog);
			printf(_("Color temperature: %luK\n"), colour.temperature);
			printf(_("Brightness: %.2f\n"), colour.brightness);
		}

		if (options.mode == PROGRAM_MODE_PRINT)
			break;

	apply:
		if (options.method->set_temperature(method_state, &colour, options.preserve_gamma) < 0)
			eprintf(_("Temperature adjustment failed."));

#ifndef WINDOWS
		/* In Quartz (OSX) the gamma adjustments will automatically revert when
		 * the process exits. Therefore, we have to loop until CTRL-C is received. */
		if (!strcmp(options.method->name, "quartz")) {
			weprintf(_("Press ctrl-c to stop..."));
			while (!exiting)
				pause();
		}
#endif
		break;

	case PROGRAM_MODE_MANUAL:
		if (options.verbose)
			printf(_("Color temperature: %luK\n"), options.temp_set);
		colour = scheme->day;
		colour.temperature = options.temp_set;
		goto apply;

	case PROGRAM_MODE_RESET:
		colour = COLOUR_SETTING_NEUTRAL;
		options.preserve_gamma = 0;
		goto apply;

	case PROGRAM_MODE_CONTINUAL:
		run_continual_mode(options.provider, location_state, scheme, options.method, method_state,
		                   options.use_fade, options.preserve_gamma, options.verbose);
		break;
	}

	if (options.mode != PROGRAM_MODE_PRINT)
		options.method->free(method_state);
	if (need_location)
		options.provider->free(location_state);
	return 0;
}