9ae3a8
/* Start/stop KSM, for systemd.
9ae3a8
 * Copyright (C) 2009, 2011 Red Hat, Inc.
9ae3a8
 * Written by Paolo Bonzini <pbonzini@redhat.com>.
9ae3a8
 * Based on the original sysvinit script by Dan Kenigsberg <danken@redhat.com>
9ae3a8
 * This file is distributed under the GNU General Public License, version 2
9ae3a8
 * or later.  */
9ae3a8
9ae3a8
#include <unistd.h>
9ae3a8
#include <stdio.h>
9ae3a8
#include <limits.h>
9ae3a8
#include <stdint.h>
9ae3a8
#include <stdlib.h>
9ae3a8
#include <string.h>
9ae3a8
9ae3a8
#define KSM_MAX_KERNEL_PAGES_FILE "/sys/kernel/mm/ksm/max_kernel_pages"
9ae3a8
#define KSM_RUN_FILE		  "/sys/kernel/mm/ksm/run"
9ae3a8
9ae3a8
char *program_name;
9ae3a8
9ae3a8
int usage(void)
9ae3a8
{
9ae3a8
	fprintf(stderr, "Usage: %s {start|stop}\n", program_name);
9ae3a8
	return 1;
9ae3a8
}
9ae3a8
9ae3a8
int write_value(uint64_t value, char *filename)
9ae3a8
{
9ae3a8
	FILE *fp;
9ae3a8
	if (!(fp = fopen(filename, "w")) ||
9ae3a8
	    fprintf(fp, "%llu\n", (unsigned long long) value) == EOF ||
9ae3a8
	    fflush(fp) == EOF ||
9ae3a8
	    fclose(fp) == EOF)
9ae3a8
		return 1;
9ae3a8
9ae3a8
	return 0;
9ae3a8
}
9ae3a8
9ae3a8
uint64_t ksm_max_kernel_pages()
9ae3a8
{
9ae3a8
	char *var = getenv("KSM_MAX_KERNEL_PAGES");
9ae3a8
	char *endptr;
9ae3a8
	uint64_t value;
9ae3a8
	if (var && *var) {
9ae3a8
		value = strtoll(var, &endptr, 0);
9ae3a8
		if (value < LLONG_MAX && !*endptr)
9ae3a8
			return value;
9ae3a8
	}
9ae3a8
	/* Unless KSM_MAX_KERNEL_PAGES is set, let KSM munch up to half of
9ae3a8
	 * total memory.  */
9ae3a8
	return sysconf(_SC_PHYS_PAGES) / 2;
9ae3a8
}
9ae3a8
9ae3a8
int start(void)
9ae3a8
{
9ae3a8
	if (access(KSM_MAX_KERNEL_PAGES_FILE, R_OK) >= 0)
9ae3a8
		write_value(ksm_max_kernel_pages(), KSM_MAX_KERNEL_PAGES_FILE);
9ae3a8
	return write_value(1, KSM_RUN_FILE);
9ae3a8
}
9ae3a8
9ae3a8
int stop(void)
9ae3a8
{
9ae3a8
	return write_value(0, KSM_RUN_FILE);
9ae3a8
}
9ae3a8
9ae3a8
int main(int argc, char **argv)
9ae3a8
{
9ae3a8
	program_name = argv[0];
9ae3a8
	if (argc < 2) {
9ae3a8
		return usage();
9ae3a8
	} else if (!strcmp(argv[1], "start")) {
9ae3a8
		return start();
9ae3a8
	} else if (!strcmp(argv[1], "stop")) {
9ae3a8
		return stop();
9ae3a8
	} else {
9ae3a8
		return usage();
9ae3a8
	}
9ae3a8
}