/* * Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana * University Research and Technology * Corporation. All rights reserved. * Copyright (c) 2004-2005 The University of Tennessee and The University * of Tennessee Research Foundation. All rights * reserved. * Copyright (c) 2004-2005 High Performance Computing Center Stuttgart, * University of Stuttgart. All rights reserved. * Copyright (c) 2004-2005 The Regents of the University of California. * All rights reserved. * Copyright (c) 2006-2007 Cisco Systems, Inc. All rights reserved. * * $COPYRIGHT$ * * Additional copyrights may follow * * $HEADER$ */ #include "opal_config.h" /* This component will only be compiled on Linux, where we are guaranteed to have and friends */ #include #include #include #include #include #include "opal/constants.h" #include "opal/mca/base/mca_base_param.h" #include "opal/mca/paffinity/paffinity.h" #include "opal/mca/paffinity/base/base.h" #include "paffinity_linux.h" #include "plpa/src/libplpa/plpa.h" /* * Local functions */ static int linux_module_init(void); static int linux_module_set(opal_paffinity_base_cpu_set_t cpumask); static int linux_module_get(opal_paffinity_base_cpu_set_t *cpumask); /* * Linux paffinity module */ static const opal_paffinity_base_module_1_1_0_t module = { /* Initialization function */ linux_module_init, /* Module function pointers */ linux_module_set, linux_module_get }; const opal_paffinity_base_module_1_1_0_t * opal_paffinity_linux_component_query(int *query) { int param; param = mca_base_param_find("paffinity", "linux", "priority"); mca_base_param_lookup_int(param, query); return &module; } static int linux_module_init(void) { /* Nothing to do */ return OPAL_SUCCESS; } /************************************************************************ See the note in paffinity_linux.h -- there are at least 3 different ways that Linux's sched_setaffinity()/sched_getaffinity() are implemented. Thankfully there is the Portable Linux Processor Affinity project which determines the flavor of affinity at runtime and takes care of of the problem. Using get/set affinity functions from plpa - configured with an opal prefix. User needs to set a mask with the bit number of the cpu set. We provide macros to do this. ************************************************************************/ static int linux_module_set(opal_paffinity_base_cpu_set_t mask) { opal_paffinity_linux_plpa_cpu_set_t plpa_mask; unsigned int i; if (sizeof(mask) > sizeof(plpa_mask)) { return OPAL_ERR_BAD_PARAM; } else { PLPA_CPU_ZERO(&plpa_mask); for (i = 0; i < sizeof(plpa_mask) ; i++) { if (PLPA_CPU_ISSET(i,&mask)) { PLPA_CPU_SET(i,&plpa_mask); } } } if (0 != opal_paffinity_linux_plpa_sched_setaffinity(getpid(), sizeof(plpa_mask), &plpa_mask)) { return OPAL_ERR_IN_ERRNO; } return OPAL_SUCCESS; } static int linux_module_get(opal_paffinity_base_cpu_set_t *mask) { opal_paffinity_linux_plpa_cpu_set_t plpa_mask; unsigned int i; if (NULL == mask) { return OPAL_ERR_BAD_PARAM; } if (sizeof(*mask) > sizeof(plpa_mask)) { return OPAL_ERR_BAD_PARAM; /* look up in header file */ } if (0 != opal_paffinity_linux_plpa_sched_getaffinity(getpid(), sizeof(plpa_mask), &plpa_mask)) { return OPAL_ERR_IN_ERRNO; } for (i = 0; i < sizeof(mask); i++) { if (PLPA_CPU_ISSET(i,&plpa_mask)) { PLPA_CPU_SET(i,mask); } } return OPAL_SUCCESS; }