#include #include #include #include /* for cpu sets */ #include #include #include "migration.h" extern ssize_t read_file(const char* fname, void* buf, size_t maxlen); int release_master() { static const char NO_CPU[] = "NO_CPU"; char buf[7] = {0}; /* up to 999999 CPUs */ int master = -1; int ret = read_file("/proc/litmus/release_master", &buf, sizeof(buf)-1); if ((ret > 0) && (strncmp(buf, NO_CPU, sizeof(NO_CPU)-1) != 0)) master = atoi(buf); return master; } int num_online_cpus() { return sysconf(_SC_NPROCESSORS_ONLN); } int ludwig_vcpu_to_pcpu(int vcpu) { static const int LUDWIG_VCPU_TO_PCPU[] = { 0, 4, 8, 12, 16, 20, 1, 5, 9, 13, 17, 21, 2, 6, 10, 14, 18, 22, 3, 7, 11, 15, 19, 23 }; assert(vcpu < 24 && vcpu >= 0); return LUDWIG_VCPU_TO_PCPU[vcpu]; } int ludwig_pcpu_to_vcpu(int pcpu) { static const int LUDWIG_PCPU_TO_VCPU[] = { 0, 6, 12, 18, 1, 7, 13, 19, 2, 8, 14, 20, 3, 9, 15, 21, 4, 10, 16, 22, 5, 11, 17, 23 }; assert(pcpu < 24 && pcpu >= 0); return LUDWIG_PCPU_TO_VCPU[pcpu]; } int partition_to_cpu(int partition) { int cpu = partition; int master = release_master(); if (master != -1 && master <= cpu) { ++cpu; /* skip over the release master */ } return ludwig_vcpu_to_pcpu(cpu); } int cluster_to_first_cpu(int cluster, int cluster_sz) { int first_cpu; int master; if (cluster_sz == 1) return partition_to_cpu(cluster); master = release_master(); first_cpu = cluster * cluster_sz; if (master == first_cpu) ++first_cpu; return ludwig_vcpu_to_pcpu(first_cpu); } int be_migrate_thread_to_cpu(pid_t tid, int target_pcpu) { cpu_set_t *cpu_set; size_t sz; int num_cpus; int ret; /* TODO: Error check to make sure that tid is not a real-time task. */ if (target_pcpu < 0) return -1; num_cpus = num_online_cpus(); if (num_cpus == -1) return -1; if (target_pcpu >= num_cpus) return -1; cpu_set = CPU_ALLOC(num_cpus); sz = CPU_ALLOC_SIZE(num_cpus); CPU_ZERO_S(sz, cpu_set); CPU_SET_S(target_pcpu, sz, cpu_set); /* apply to caller */ if (tid == 0) tid = gettid(); ret = sched_setaffinity(tid, sz, cpu_set); CPU_FREE(cpu_set); return ret; } int be_migrate_thread_to_cluster(pid_t tid, int cluster, int cluster_sz) { return __be_migrate_thread_to_cluster(tid, cluster, cluster_sz, 0); } int __be_migrate_thread_to_cluster(pid_t tid, int cluster, int cluster_sz, int ignore_rm) { int first_cpu = cluster * cluster_sz; /* first CPU in cluster */ int last_cpu = first_cpu + cluster_sz - 1; int master; int num_cpus; cpu_set_t *cpu_set; size_t sz; int i; int ret; /* TODO: Error check to make sure that tid is not a real-time task. */ if (cluster_sz == 1) { /* we're partitioned */ return be_migrate_thread_to_partition(tid, cluster); } master = (ignore_rm) ? -1 : release_master(); num_cpus = num_online_cpus(); if (num_cpus == -1 || last_cpu >= num_cpus || first_cpu < 0) return -1; cpu_set = CPU_ALLOC(num_cpus); sz = CPU_ALLOC_SIZE(num_cpus); CPU_ZERO_S(sz, cpu_set); for (i = first_cpu; i <= last_cpu; ++i) { if (i != master) { int pcpu = ludwig_vcpu_to_pcpu(i); CPU_SET_S(pcpu, sz, cpu_set); } } /* apply to caller */ if (tid == 0) tid = gettid(); ret = sched_setaffinity(tid, sz, cpu_set); CPU_FREE(cpu_set); return ret; } int be_migrate_thread_to_partition(pid_t tid, int partition) { return be_migrate_thread_to_cpu(tid, partition_to_cpu(partition)); } int be_migrate_to_cpu(int target_pcpu) { return be_migrate_thread_to_cpu(0, target_pcpu); } int be_migrate_to_cluster(int cluster, int cluster_sz) { return be_migrate_thread_to_cluster(0, cluster, cluster_sz); } int be_migrate_to_partition(int partition) { return be_migrate_thread_to_partition(0, partition); }