/* -*- Mode: C; c-basic-offset:4 ; -*- */ /* * (C) 2001 by Argonne National Laboratory. * See COPYRIGHT in top-level directory. */ /* Allow fprintf to logfile */ /* style: allow:fprintf:1 sig:0 */ /* Utility functions associated with PMI implementation, but not part of the PMI interface itself. Reading and writing on pipes, signals, and parsing key=value messages */ #include #include #include #include #include #include "pmi2_util.h" #define MAXVALLEN 1024 #define MAXKEYLEN 32 /* These are not the keyvals in the keyval space that is part of the PMI specification. They are just part of this implementation's internal utilities. */ struct PMI2U_keyval_pairs { char key[MAXKEYLEN]; char value[MAXVALLEN]; }; static struct PMI2U_keyval_pairs PMI2U_keyval_tab[64] = { { { 0 }, { 0 } } }; static int PMI2U_keyval_tab_idx = 0; /* This is used to prepend printed output. Set the initial value to "unset" */ static char PMI2U_print_id[PMI2_IDSIZE] = "unset"; void PMI2U_Set_rank(int PMI_rank) { snprintf(PMI2U_print_id, PMI2_IDSIZE, "cli_%d", PMI_rank); } void PMI2U_SetServer(void) { strncpy(PMI2U_print_id, "server", PMI2_IDSIZE); } #define MAX_READLINE 1024 /* * Return the next newline-terminated string of maximum length maxlen. * This is a buffered version, and reads from fd as necessary. A */ int PMI2U_readline(int fd, char *buf, int maxlen) { static char readbuf[MAX_READLINE]; static char *nextChar = 0, *lastChar = 0; /* lastChar is really one past last char */ int curlen, n; char *p, ch; /* Note: On the client side, only one thread at a time should be calling this, and there should only be a single fd. Server side code should not use this routine (see the replacement version in src/pm/util/pmiserv.c) */ /*PMI2U_Assert(nextChar == lastChar || fd == lastfd);*/ p = buf; curlen = 1; /* Make room for the null */ while (curlen < maxlen) { if (nextChar == lastChar) { do { n = read(fd, readbuf, sizeof(readbuf) - 1); } while (n == -1 && errno == EINTR); if (n == 0) { /* EOF */ break; } else if (n < 0) { if (curlen == 1) { curlen = 0; } break; } nextChar = readbuf; lastChar = readbuf + n; /* Add a null at the end just to make it easier to print the read buffer */ readbuf[n] = 0; /* FIXME: Make this an optional output */ /* printf( "Readline %s\n", readbuf ); */ } ch = *nextChar++; *p++ = ch; curlen++; if (ch == '\n') break; } /* We null terminate the string for convenience in printing */ *p = 0; PMI2U_printf("PMI received: %s", buf); /* Return the number of characters, not counting the null */ return curlen - 1; } int PMI2U_writeline(int fd, char *buf) { int size = strlen(buf), n; if (buf[size - 1] != '\n') /* error: no newline at end */ PMI2U_printf("write_line: message string doesn't end in newline: :%s:", buf); else { PMI2U_printf("PMI sending: %s", buf); do { n = write(fd, buf, size); } while (n == -1 && errno == EINTR); if (n < 0) { PMI2U_printf("write_line error; fd=%d buf=:%s:", fd, buf); return (-1); } if (n < size) PMI2U_printf("write_line failed to write entire message"); } return 0; } /* * Given an input string st, parse it into internal storage that can be * queried by routines such as PMI2U_getval. */ int PMI2U_parse_keyvals(char *st) { char *p, *keystart, *valstart; int offset; if (!st) return (-1); PMI2U_keyval_tab_idx = 0; p = st; while (1) { while (*p == ' ') p++; /* got non-blank */ if (*p == '=') { PMI2U_printf("PMI2U_parse_keyvals: unexpected = at character %ld in %s", (long int) (p - st), st); return (-1); } if (*p == '\n' || *p == '\0') return (0); /* normal exit */ /* got normal character */ keystart = p; /* remember where key started */ while (*p != ' ' && *p != '=' && *p != '\n' && *p != '\0') p++; if (*p == ' ' || *p == '\n' || *p == '\0') { PMI2U_printf("PMI2U_parse_keyvals: unexpected key delimiter at character %ld in %s", (long int) (p - st), st); return (-1); } /* Null terminate the key */ *p = 0; /* store key */ strncpy(PMI2U_keyval_tab[PMI2U_keyval_tab_idx].key, keystart, MAXKEYLEN); PMI2U_keyval_tab[PMI2U_keyval_tab_idx].key[MAXKEYLEN-1] = '\0'; valstart = ++p; /* start of value */ while (*p != ' ' && *p != '\n' && *p != '\0') p++; /* store value */ strncpy(PMI2U_keyval_tab[PMI2U_keyval_tab_idx].value, valstart, MAXVALLEN); offset = p - valstart; /* When compiled with -fPIC, the pgcc compiler generates incorrect code if "p - valstart" is used instead of using the intermediate offset */ PMI2U_keyval_tab[PMI2U_keyval_tab_idx].value[offset] = '\0'; PMI2U_keyval_tab_idx++; if (*p == ' ') continue; if (*p == '\n' || *p == '\0') return (0); /* value has been set to empty */ } } void PMI2U_dump_keyvals(void) { int i; for (i = 0; i < PMI2U_keyval_tab_idx; i++) PMI2U_printf(" %s=%s", PMI2U_keyval_tab[i].key, PMI2U_keyval_tab[i].value); } char *PMI2U_getval(const char *keystr, char *valstr, int vallen) { int i; for (i = 0; i < PMI2U_keyval_tab_idx; i++) { if (strcmp(keystr, PMI2U_keyval_tab[i].key) == 0) { MPIU_Strncpy(valstr, PMI2U_keyval_tab[i].value, vallen); PMI2U_keyval_tab[i].value[vallen-1] = '\0'; return valstr; } } valstr[0] = '\0'; return NULL ; } void PMI2U_chgval(const char *keystr, char *valstr) { int i; for (i = 0; i < PMI2U_keyval_tab_idx; i++) { if (strcmp(keystr, PMI2U_keyval_tab[i].key) == 0) { strncpy(PMI2U_keyval_tab[i].value, valstr, MAXVALLEN); PMI2U_keyval_tab[i].value[MAXVALLEN - 1] = '\0'; } } } /* This code is borrowed from mpich2-1.5/src/pm/util/safestr2.c. The reason is to keep the save code logic around strncpy() as as in the original PMI2 implementation. @ MPIU_Strncpy - Copy a string with a maximum length Input Parameters: + instr - String to copy - maxlen - Maximum total length of 'outstr' Output Parameter: . outstr - String to copy into Notes: This routine is the routine that you wish 'strncpy' was. In copying 'instr' to 'outstr', it stops when either the end of 'outstr' (the null character) is seen or the maximum length 'maxlen' is reached. Unlike 'strncpy', it does not add enough nulls to 'outstr' after copying 'instr' in order to move precisely 'maxlen' characters. Thus, this routine may be used anywhere 'strcpy' is used, without any performance cost related to large values of 'maxlen'. If there is insufficient space in the destination, the destination is still null-terminated, to avoid potential failures in routines that neglect to check the error code return from this routine. Module: Utility @*/ int MPIU_Strncpy(char *dest, const char *src, size_t n) { char *d_ptr = dest; const char *s_ptr = src; register int i; if (n == 0) return 0; i = (int)n; while (*s_ptr && i-- > 0) { *d_ptr++ = *s_ptr++; } if (i > 0) { *d_ptr = 0; return 0; } else { /* Force a null at the end of the string (gives better safety in case the user fails to check the error code) */ dest[n-1] = 0; /* We may want to force an error message here, at least in the debugging version */ /* printf( "failure in copying %s with length %d\n", src, n ); */ return 1; } }