src/base/Makefile.am        |   16 +-
 src/base/hash.cc            |  239 ++++++++++++++++++++++++++++++++++++++++++++
 src/base/hash.h             |   32 +++++
 src/base/tests/hash_test.cc |  109 ++++++++++++++++++++
 4 files changed, 390 insertions(+), 6 deletions(-)


Add a function that returns a 32 character string which is generated by
computing the SHA-512/192 hash of the message parameter, and encoding the result
using Base64 with a URL and filename safe alphabet. For more information about
these algorithms, refer to the corresponding standards. SHA-512/192 is defined
in FIPS 180-4. Base64 is defined in RFC 4648.

diff --git a/src/base/Makefile.am b/src/base/Makefile.am
--- a/src/base/Makefile.am
+++ b/src/base/Makefile.am
@@ -35,6 +35,7 @@ lib_libopensaf_core_la_SOURCES += \
        src/base/daemon.c \
        src/base/file_notify.cc \
        src/base/getenv.cc \
+       src/base/hash.cc \
        src/base/hj_dec.c \
        src/base/hj_edp.c \
        src/base/hj_edu.c \
@@ -79,6 +80,7 @@ noinst_HEADERS += \
        src/base/daemon.h \
        src/base/file_notify.h \
        src/base/getenv.h \
+       src/base/hash.h \
        src/base/log_message.h \
        src/base/logtrace.h \
        src/base/macros.h \
@@ -176,16 +178,18 @@ bin_libbase_test_CPPFLAGS = \
 
 bin_libbase_test_LDFLAGS = \
        $(AM_LDFLAGS) \
-       src/base/lib_libopensaf_core_la-unix_socket.lo \
-       src/base/lib_libopensaf_core_la-unix_server_socket.lo \
+       src/base/lib_libopensaf_core_la-file_notify.lo \
        src/base/lib_libopensaf_core_la-getenv.lo \
+       src/base/lib_libopensaf_core_la-hash.lo \
        src/base/lib_libopensaf_core_la-log_message.lo \
        src/base/lib_libopensaf_core_la-process.lo \
-       src/base/lib_libopensaf_core_la-file_notify.lo
+       src/base/lib_libopensaf_core_la-unix_server_socket.lo \
+       src/base/lib_libopensaf_core_la-unix_socket.lo
 
 bin_libbase_test_SOURCES = \
-       src/base/tests/unix_socket_test.cc \
+       src/base/tests/file_notify_test.cc \
        src/base/tests/getenv_test.cc \
+       src/base/tests/hash_test.cc \
        src/base/tests/log_message_test.cc \
        src/base/tests/mock_logtrace.cc \
        src/base/tests/mock_osaf_abort.cc \
@@ -193,8 +197,8 @@ bin_libbase_test_SOURCES = \
        src/base/tests/time_add_test.cc \
        src/base/tests/time_compare_test.cc \
        src/base/tests/time_convert_test.cc \
-       src/base/tests/file_notify_test.cc \
-       src/base/tests/time_subtract_test.cc
+       src/base/tests/time_subtract_test.cc \
+       src/base/tests/unix_socket_test.cc
 
 bin_libbase_test_LDADD = \
        $(GTEST_DIR)/lib/libgtest.la \
diff --git a/src/base/hash.cc b/src/base/hash.cc
new file mode 100644
--- /dev/null
+++ b/src/base/hash.cc
@@ -0,0 +1,239 @@
+/*      -*- OpenSAF  -*-
+ *
+ * Copyright Ericsson AB 2017 - All Rights Reserved.
+ *
+ * This program 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. This file and program are licensed
+ * under the GNU Lesser General Public License Version 2.1, February 1999.
+ * The complete license can be accessed from the following location:
+ * http://opensource.org/licenses/lgpl-license.php
+ * See the Copying file included with the OpenSAF distribution for full
+ * licensing terms.
+ *
+ */
+
+#include "base/hash.h"
+#include <endian.h>
+#include <cassert>
+#include <cstdint>
+
+namespace base {
+
+namespace {
+
+constexpr uint64_t K[] = {
+  UINT64_C(0x428a2f98d728ae22), UINT64_C(0x7137449123ef65cd),
+  UINT64_C(0xb5c0fbcfec4d3b2f), UINT64_C(0xe9b5dba58189dbbc),
+  UINT64_C(0x3956c25bf348b538), UINT64_C(0x59f111f1b605d019),
+  UINT64_C(0x923f82a4af194f9b), UINT64_C(0xab1c5ed5da6d8118),
+  UINT64_C(0xd807aa98a3030242), UINT64_C(0x12835b0145706fbe),
+  UINT64_C(0x243185be4ee4b28c), UINT64_C(0x550c7dc3d5ffb4e2),
+  UINT64_C(0x72be5d74f27b896f), UINT64_C(0x80deb1fe3b1696b1),
+  UINT64_C(0x9bdc06a725c71235), UINT64_C(0xc19bf174cf692694),
+  UINT64_C(0xe49b69c19ef14ad2), UINT64_C(0xefbe4786384f25e3),
+  UINT64_C(0x0fc19dc68b8cd5b5), UINT64_C(0x240ca1cc77ac9c65),
+  UINT64_C(0x2de92c6f592b0275), UINT64_C(0x4a7484aa6ea6e483),
+  UINT64_C(0x5cb0a9dcbd41fbd4), UINT64_C(0x76f988da831153b5),
+  UINT64_C(0x983e5152ee66dfab), UINT64_C(0xa831c66d2db43210),
+  UINT64_C(0xb00327c898fb213f), UINT64_C(0xbf597fc7beef0ee4),
+  UINT64_C(0xc6e00bf33da88fc2), UINT64_C(0xd5a79147930aa725),
+  UINT64_C(0x06ca6351e003826f), UINT64_C(0x142929670a0e6e70),
+  UINT64_C(0x27b70a8546d22ffc), UINT64_C(0x2e1b21385c26c926),
+  UINT64_C(0x4d2c6dfc5ac42aed), UINT64_C(0x53380d139d95b3df),
+  UINT64_C(0x650a73548baf63de), UINT64_C(0x766a0abb3c77b2a8),
+  UINT64_C(0x81c2c92e47edaee6), UINT64_C(0x92722c851482353b),
+  UINT64_C(0xa2bfe8a14cf10364), UINT64_C(0xa81a664bbc423001),
+  UINT64_C(0xc24b8b70d0f89791), UINT64_C(0xc76c51a30654be30),
+  UINT64_C(0xd192e819d6ef5218), UINT64_C(0xd69906245565a910),
+  UINT64_C(0xf40e35855771202a), UINT64_C(0x106aa07032bbd1b8),
+  UINT64_C(0x19a4c116b8d2d0c8), UINT64_C(0x1e376c085141ab53),
+  UINT64_C(0x2748774cdf8eeb99), UINT64_C(0x34b0bcb5e19b48a8),
+  UINT64_C(0x391c0cb3c5c95a63), UINT64_C(0x4ed8aa4ae3418acb),
+  UINT64_C(0x5b9cca4f7763e373), UINT64_C(0x682e6ff3d6b2b8a3),
+  UINT64_C(0x748f82ee5defb2fc), UINT64_C(0x78a5636f43172f60),
+  UINT64_C(0x84c87814a1f0ab72), UINT64_C(0x8cc702081a6439ec),
+  UINT64_C(0x90befffa23631e28), UINT64_C(0xa4506cebde82bde9),
+  UINT64_C(0xbef9a3f7b2c67915), UINT64_C(0xc67178f2e372532b),
+  UINT64_C(0xca273eceea26619c), UINT64_C(0xd186b8c721c0c207),
+  UINT64_C(0xeada7dd6cde0eb1e), UINT64_C(0xf57d4f7fee6ed178),
+  UINT64_C(0x06f067aa72176fba), UINT64_C(0x0a637dc5a2c898a6),
+  UINT64_C(0x113f9804bef90dae), UINT64_C(0x1b710b35131c471b),
+  UINT64_C(0x28db77f523047d84), UINT64_C(0x32caab7b40c72493),
+  UINT64_C(0x3c9ebe0a15c9bebc), UINT64_C(0x431d67c49c100d4c),
+  UINT64_C(0x4cc5d4becb3e42b6), UINT64_C(0x597f299cfc657e2a),
+  UINT64_C(0x5fcb6fab3ad6faec), UINT64_C(0x6c44198c4a475817)
+};
+
+constexpr inline uint64_t ROTR(int n, uint64_t x) {
+  return (x >> n) | (x << (64 - n));
+}
+
+constexpr inline uint64_t Ch(uint64_t x, uint64_t y, uint64_t z) {
+  return (x & y) ^ (~x & z);
+}
+
+constexpr inline uint64_t Maj(uint64_t x, uint64_t y, uint64_t z) {
+  return (x & y) ^ (x & z) ^ (y & z);
+}
+
+constexpr inline uint64_t SIGMA_0(uint64_t x) {
+  return ROTR(28, x) ^ ROTR(34, x) ^ ROTR(39, x);
+}
+
+constexpr inline uint64_t SIGMA_1(uint64_t x) {
+  return ROTR(14, x) ^ ROTR(18, x) ^ ROTR(41, x);
+}
+
+constexpr inline uint64_t sigma_0(uint64_t x) {
+  return ROTR(1, x) ^ ROTR(8, x) ^ (x >> 7);
+}
+
+constexpr inline uint64_t sigma_1(uint64_t x) {
+  return ROTR(19, x) ^ ROTR(61, x) ^ (x >> 6);
+}
+
+std::string CreatePadding(uint64_t message_size_in_bytes) {
+  assert(message_size_in_bytes < (uint64_t{1} << (64 - 3)));
+  uint64_t l = 8 * message_size_in_bytes;
+  uint64_t k =
+      ((((l + 1) % 1024) <= 896) ? 896 : (1024 + 896)) - ((l + 1) % 1024);
+  assert(((l + 1 + k + 128) % 1024) == 0);
+  std::string padding;
+  padding.reserve((1 + k + 128) / 8);
+  padding.push_back(static_cast<char>(0x80));
+  padding.resize((1 + k + 64) / 8, static_cast<char>(0));
+  for (int i = 56; i >= 0; i = i - 8) {
+    padding.push_back(static_cast<char>((l >> i) & 0xffu));
+  }
+  return padding;
+}
+
+inline uint64_t MessageByte(const std::string& msg, const std::string& padding,
+                            uint64_t index) {
+  assert(index < (msg.size() + padding.size()));
+  return index < msg.size() ? static_cast<uint8_t>(msg[index]) :
+      static_cast<uint8_t>(padding[index - msg.size()]);
+}
+
+inline uint64_t M(const std::string& msg, const std::string& padding,
+                  uint64_t t, uint64_t i) {
+  assert(t <= 15 && i != 0);
+  uint64_t index = (i - 1) * 128 + t * 8;
+  return ((index + 7) < msg.size()) ?
+      be64toh(*reinterpret_cast<const uint64_t*>(msg.data() + index)) :
+      ((MessageByte(msg, padding, index + 0) << 56) |
+       (MessageByte(msg, padding, index + 1) << 48) |
+       (MessageByte(msg, padding, index + 2) << 40) |
+       (MessageByte(msg, padding, index + 3) << 32) |
+       (MessageByte(msg, padding, index + 4) << 24) |
+       (MessageByte(msg, padding, index + 5) << 16) |
+       (MessageByte(msg, padding, index + 6) << 8) |
+       (MessageByte(msg, padding, index + 7) << 0));
+}
+
+class Base64 {
+ public:
+  static std::string Encode(uint64_t H0, uint64_t H1, uint64_t H2) {
+    Base64 encoder;
+    encoder.encoded_.reserve(32);
+    encoder.Append192Bits(H0, H1, H2);
+    return encoder.encoded_;
+  }
+
+ private:
+  static constexpr uint64_t kMask6Bits = (UINT64_C(1) << 6) - 1;
+  static constexpr uint64_t kMask8Bits = (UINT64_C(1) << 8) - 1;
+  static constexpr uint64_t kMask16Bits = (UINT64_C(1) << 16) - 1;
+  static constexpr uint64_t kMask24Bits = (UINT64_C(1) << 24) - 1;
+  static constexpr char kAlphabet[64] = {
+    'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
+    'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd',
+    'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's',
+    't', 'u', 'v', 'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7',
+    '8', '9', '-', '_'
+  };
+
+  void Append6Bits(uint64_t a) {
+    assert(a < (UINT64_C(1) << 6));
+    encoded_.push_back(kAlphabet[a]);
+  }
+
+  void Append24Bits(uint64_t a) {
+    assert(a < (UINT64_C(1) << 24));
+    Append6Bits(a >> 18);
+    Append6Bits((a >> 12) & kMask6Bits);
+    Append6Bits((a >> 6) & kMask6Bits);
+    Append6Bits(a & kMask6Bits);
+  }
+
+  void Append192Bits(uint64_t H0, uint64_t H1, uint64_t H2) {
+    Append24Bits(H0 >> 40);
+    Append24Bits((H0 >> 16) & kMask24Bits);
+    Append24Bits(((H0 & kMask16Bits) << 8) | (H1 >> 56));
+    Append24Bits((H1 >> 32) & kMask24Bits);
+    Append24Bits((H1 >> 8) & kMask24Bits);
+    Append24Bits(((H1 & kMask8Bits) << 16) | (H2 >> 48));
+    Append24Bits((H2 >> 24) & kMask24Bits);
+    Append24Bits(H2 & kMask24Bits);
+  }
+
+  std::string encoded_{};
+};
+
+constexpr char Base64::kAlphabet[64];
+
+}  // namespace
+
+std::string Hash(const std::string& message) {
+  uint64_t H0 = UINT64_C(0x010176140648b233);
+  uint64_t H1 = UINT64_C(0xdb92aeb1eebadd6f);
+  uint64_t H2 = UINT64_C(0x83a9e27aa1d5ea62);
+  uint64_t H3 = UINT64_C(0xec95f77eb609b4e1);
+  uint64_t H4 = UINT64_C(0x71a99185c75caefa);
+  uint64_t H5 = UINT64_C(0x006e8f08baf32e3c);
+  uint64_t H6 = UINT64_C(0x6a2b21abd2db2aec);
+  uint64_t H7 = UINT64_C(0x24926cdbd918a27f);
+
+  std::string padding = CreatePadding(message.size());
+  uint64_t N = (message.size() + padding.size()) / 128;
+  for (uint64_t i = 1; i <= N; ++i) {
+    uint64_t W[80];
+    uint64_t a = H0;
+    uint64_t b = H1;
+    uint64_t c = H2;
+    uint64_t d = H3;
+    uint64_t e = H4;
+    uint64_t f = H5;
+    uint64_t g = H6;
+    uint64_t h = H7;
+    for (uint64_t t = 0; t <= 79; ++t) {
+      if (t <= 15) {
+        W[t] = M(message, padding, t, i);
+      } else {
+        W[t] = sigma_1(W[t - 2]) + W[t - 7] + sigma_0(W[t - 15]) + W[t - 16];
+      }
+      uint64_t T1 = h + SIGMA_1(e) + Ch(e, f, g) + K[t] + W[t];
+      uint64_t T2 = SIGMA_0(a) + Maj(a, b, c);
+      h = g;
+      g = f;
+      f = e;
+      e = d + T1;
+      d = c;
+      c = b;
+      b = a;
+      a = T1 + T2;
+    }
+    H0 += a;
+    H1 += b;
+    H2 += c;
+    H3 += d;
+    H4 += e;
+    H5 += f;
+    H6 += g;
+    H7 += h;
+  }
+  return Base64::Encode(H0, H1, H2);
+}
+
+}  // namespace base
diff --git a/src/base/hash.h b/src/base/hash.h
new file mode 100644
--- /dev/null
+++ b/src/base/hash.h
@@ -0,0 +1,32 @@
+/*      -*- OpenSAF  -*-
+ *
+ * Copyright Ericsson AB 2017 - All Rights Reserved.
+ *
+ * This program 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. This file and program are licensed
+ * under the GNU Lesser General Public License Version 2.1, February 1999.
+ * The complete license can be accessed from the following location:
+ * http://opensource.org/licenses/lgpl-license.php
+ * See the Copying file included with the OpenSAF distribution for full
+ * licensing terms.
+ *
+ */
+
+#ifndef BASE_HASH_H_
+#define BASE_HASH_H_
+
+#include <string>
+
+namespace base {
+
+// Returns a 32 character string which is generated by computing the 
SHA-512/192
+// hash of the message parameter, and encoding the result using Base64 with a
+// URL and filename safe alphabet. For more information about these algorithms,
+// refer to the corresponding standards. SHA-512/192 is defined in FIPS
+// 180-4. Base64 is defined in RFC 4648.
+std::string Hash(const std::string& message);
+
+}  // namespace base
+
+#endif  // BASE_HASH_H_
diff --git a/src/base/tests/hash_test.cc b/src/base/tests/hash_test.cc
new file mode 100644
--- /dev/null
+++ b/src/base/tests/hash_test.cc
@@ -0,0 +1,109 @@
+/*      -*- OpenSAF  -*-
+ *
+ * Copyright Ericsson AB 2017 - All Rights Reserved.
+ *
+ * This program 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. This file and program are licensed
+ * under the GNU Lesser General Public License Version 2.1, February 1999.
+ * The complete license can be accessed from the following location:
+ * http://opensource.org/licenses/lgpl-license.php
+ * See the Copying file included with the OpenSAF distribution for full
+ * licensing terms.
+ *
+ */
+
+#include <endian.h>
+#include <cstddef>
+#include <random>
+#include <string>
+#include "base/hash.h"
+#include "gtest/gtest.h"
+
+struct TestVector {
+  const char* key_data;
+  size_t key_size;
+  const char* hash_value;
+};
+
+#define TEST_VECTOR(key, hash) { key, sizeof(key) - 1, hash }
+
+static const TestVector test_vectors[] = {
+  TEST_VECTOR("", "mJbyfHPNxOzI7KPhb27rY6_gS2wNOSds"),
+  TEST_VECTOR("\000", "U-4lHisMY1zsvi2ry5wqlYKUMhhb_MX8"),
+  TEST_VECTOR("\001", "7R0FXuGlh60oucbCoEIyp97oPDI34hNn"),
+  TEST_VECTOR("\376", "X-7TtfTMWhw7T9CK1-yk8bRAlF_544MB"),
+  TEST_VECTOR("\377", "OMGVy2Bhf3bPvd_yYKNVn8AK-EhkSJN4"),
+  TEST_VECTOR(" ", "LOR-HJ8c_bFoH7LVc45B67zrxvqNp2ih"),
+  TEST_VECTOR("\t", "jwSZSrsTHsB4kB2nUvpebPSpvDdt8d7V"),
+  TEST_VECTOR("\n", "WlgTrOkNrltdky0QZv1B3jMG0D73h3YC"),
+  TEST_VECTOR("\r\n", "pAv_qrHWSbOLxiJmXB2ETosp_FiE-G4H"),
+  TEST_VECTOR("\\/:", "wyC3ktDwB4tvyzXBO5zx_iH76GKFmjxB"),
+  TEST_VECTOR("$abc", "xSHQCvFJpVfx4ofdholD-mzgxyJJHjYg"),
+  TEST_VECTOR("\000\000\000\000\000", "mzklNJ6l6vWVe73PvthH61RRsl90ygC_"),
+  TEST_VECTOR("123456", "E_7JXvueRDhde3Wq3W5jYSsts_LOWJNZ"),
+  TEST_VECTOR("1234567", "0cAgD2NO2ASyz39axMrnBdV_VuHn2PhC"),
+  TEST_VECTOR("12345678", "Uzmag7xvTQg40VBwOumXaugZxFZyRpB8"),
+  TEST_VECTOR("123456789", "3YfQdE3kVpdmBEaBHM4uCj1feXvixjYA"),
+  
TEST_VECTOR("\200\201\202\203\204\205\206\207\210\211\212\213\214\215\216\217"
+              
"\220\221\222\223\224\225\226\227\230\231\232\233\234\235\236\237"
+              
"\240\241\242\243\244\245\246\247\250\251\252\253\254\255\256\257"
+              
"\260\261\262\263\264\265\266\267\270\271\272\273\274\275\276\277"
+              
"\300\301\302\303\304\305\306\307\310\311\312\313\314\315\316\317"
+              
"\320\321\322\323\324\325\326\327\330\331\332\333\334\335\336\337"
+              "\340\341\342\343\344\345\346\347\350\351\352\353\354\355",
+              "z2V3p2mUUigcBkYWSEw0yhz2f1v4M8Nb"),
+  
TEST_VECTOR("\200\201\202\203\204\205\206\207\210\211\212\213\214\215\216\217"
+              
"\220\221\222\223\224\225\226\227\230\231\232\233\234\235\236\237"
+              
"\240\241\242\243\244\245\246\247\250\251\252\253\254\255\256\257"
+              
"\260\261\262\263\264\265\266\267\270\271\272\273\274\275\276\277"
+              
"\300\301\302\303\304\305\306\307\310\311\312\313\314\315\316\317"
+              
"\320\321\322\323\324\325\326\327\330\331\332\333\334\335\336\337"
+              "\340\341\342\343\344\345\346\347\350\351\352\353\354\355\356",
+              "WprfN6xOSNY9RHaNCkR12yEEx5hGfwHj"),
+  
TEST_VECTOR("\200\201\202\203\204\205\206\207\210\211\212\213\214\215\216\217"
+              
"\220\221\222\223\224\225\226\227\230\231\232\233\234\235\236\237"
+              
"\240\241\242\243\244\245\246\247\250\251\252\253\254\255\256\257"
+              
"\260\261\262\263\264\265\266\267\270\271\272\273\274\275\276\277"
+              
"\300\301\302\303\304\305\306\307\310\311\312\313\314\315\316\317"
+              
"\320\321\322\323\324\325\326\327\330\331\332\333\334\335\336\337"
+              "\340\341\342\343\344\345\346\347\350\351\352\353\354\355\356"
+              "\357",
+              "K3wQd-ZX5k7_AZ7p1qOwrehV5vkaJq9E"),
+  
TEST_VECTOR("\200\201\202\203\204\205\206\207\210\211\212\213\214\215\216\217"
+              
"\220\221\222\223\224\225\226\227\230\231\232\233\234\235\236\237"
+              
"\240\241\242\243\244\245\246\247\250\251\252\253\254\255\256\257"
+              
"\260\261\262\263\264\265\266\267\270\271\272\273\274\275\276\277"
+              
"\300\301\302\303\304\305\306\307\310\311\312\313\314\315\316\317"
+              
"\320\321\322\323\324\325\326\327\330\331\332\333\334\335\336\337"
+              
"\340\341\342\343\344\345\346\347\350\351\352\353\354\355\356\357"
+              "\360",
+              "O-ERJdo-W-iCV4b2QogLJmtR1gMIhlkx")
+};
+
+TEST(BaseHash, TestVectors) {
+  for (size_t i = 0; i < sizeof(test_vectors) / sizeof(test_vectors[0]); ++i) {
+    std::string key = std::string(test_vectors[i].key_data,
+                                  test_vectors[i].key_size);
+    std::string hash = std::string(test_vectors[i].hash_value);
+    std::string computed_hash = base::Hash(key);
+    EXPECT_EQ(hash, computed_hash);
+  }
+}
+
+TEST(BaseHash, LongRandomString) {
+  std::string key;
+  key.reserve(8000);
+  std::mt19937_64 generator(4711);
+  for (size_t i = 0; i != 1000; ++i) {
+    union {
+      char bytes[sizeof(uint64_t)];
+      uint64_t word;
+    };
+    word = htobe64(generator());
+    key.append(bytes, sizeof(uint64_t));
+  }
+  std::string hash = std::string("C_r2XVQjnVbnpQTMHO2OQJ2sn0YZi1py");
+  std::string computed_hash = base::Hash(key);
+  EXPECT_EQ(hash, computed_hash);
+}

------------------------------------------------------------------------------
Check out the vibrant tech community on one of the world's most
engaging tech sites, SlashDot.org! http://sdm.link/slashdot
_______________________________________________
Opensaf-devel mailing list
Opensaf-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/opensaf-devel

Reply via email to