test/hotspot/gtest/metaspace/test_commitmask.cpp
author stuefe
Thu, 17 Oct 2019 16:39:45 +0200
branchstuefe-new-metaspace-branch
changeset 58683 2d5dd194c65c
parent 58099 5aeb07390c74
permissions -rw-r--r--
Lessen verification costs

/*
 * Copyright (c) 2019, SAP SE. All rights reserved.
 * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code 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.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */


#include "precompiled.hpp"
#include "runtime/os.hpp"

#include "metaspaceTestsCommon.hpp"

static int get_random(int limit) { return os::random() % limit; }

class CommitMaskTest {
  const MetaWord* const _base;
  const size_t _word_size;

  CommitMask _mask;

  void verify_mask() {
    // Note: we omit the touch test since we operate on fictional
    // memory
    DEBUG_ONLY(_mask.verify(false);)
  }

  // Return a random sub range within [_base.._base + word_size),
  // aligned to granule size
  const MetaWord* calc_random_subrange(size_t* p_word_size) {
    size_t l1 = get_random((int)_word_size);
    size_t l2 = get_random((int)_word_size);
    if (l1 > l2) {
      size_t l = l1;
      l1 = l2;
      l2 = l;
    }
    l1 = align_down(l1, Settings::commit_granule_words());
    l2 = align_up(l2, Settings::commit_granule_words());

    const MetaWord* p = _base + l1;
    const size_t len = l2 - l1;

    assert(p >= _base && p + len <= _base + _word_size,
           "Sanity");
    *p_word_size = len;

    return p;
  }

  void test1() {

    LOG("test1");

    // Commit everything
    size_t prior_committed = _mask.mark_range_as_committed(_base, _word_size);
    verify_mask();
    ASSERT_LE(prior_committed, _word_size); // We do not really know

    // Commit everything again, should be a noop
    prior_committed = _mask.mark_range_as_committed(_base, _word_size);
    verify_mask();
    ASSERT_EQ(prior_committed, _word_size);

    ASSERT_EQ(_mask.get_committed_size(),
              _word_size);
    ASSERT_EQ(_mask.get_committed_size_in_range(_base, _word_size),
              _word_size);

    for (const MetaWord* p = _base; p < _base + _word_size; p ++) {
      ASSERT_TRUE(_mask.is_committed_address(p));
    }

    // Now make an uncommitted hole
    size_t sr_word_size;
    const MetaWord* sr_base = calc_random_subrange(&sr_word_size);
    LOG("subrange " PTR_FORMAT "-" PTR_FORMAT ".",
        p2i(sr_base), p2i(sr_base + sr_word_size));

    size_t prior_uncommitted =
        _mask.mark_range_as_uncommitted(sr_base, sr_word_size);
    verify_mask();
    ASSERT_EQ(prior_uncommitted, (size_t)0);

    // Again, for fun, should be a noop now.
    prior_uncommitted = _mask.mark_range_as_uncommitted(sr_base, sr_word_size);
    verify_mask();
    ASSERT_EQ(prior_uncommitted, sr_word_size);

    ASSERT_EQ(_mask.get_committed_size_in_range(sr_base, sr_word_size),
              (size_t)0);
    ASSERT_EQ(_mask.get_committed_size(),
              _word_size - sr_word_size);
    ASSERT_EQ(_mask.get_committed_size_in_range(_base, _word_size),
              _word_size - sr_word_size);
    for (const MetaWord* p = _base; p < _base + _word_size; p ++) {
      if (p >= sr_base && p < sr_base + sr_word_size) {
        ASSERT_FALSE(_mask.is_committed_address(p));
      } else {
        ASSERT_TRUE(_mask.is_committed_address(p));
      }
    }

    // Recommit whole range
    prior_committed = _mask.mark_range_as_committed(_base, _word_size);
    verify_mask();
    ASSERT_EQ(prior_committed, _word_size - sr_word_size);

    ASSERT_EQ(_mask.get_committed_size_in_range(sr_base, sr_word_size),
              sr_word_size);
    ASSERT_EQ(_mask.get_committed_size(),
              _word_size);
    ASSERT_EQ(_mask.get_committed_size_in_range(_base, _word_size),
              _word_size);
    for (const MetaWord* p = _base; p < _base + _word_size; p ++) {
      ASSERT_TRUE(_mask.is_committed_address(p));
    }

  }

  void test2() {

    LOG("test2");

    // Uncommit everything
    size_t prior_uncommitted = _mask.mark_range_as_uncommitted(_base, _word_size);
    verify_mask();
    ASSERT_LE(prior_uncommitted, _word_size);

    // Uncommit everything again, should be a noop
    prior_uncommitted = _mask.mark_range_as_uncommitted(_base, _word_size);
    verify_mask();
    ASSERT_EQ(prior_uncommitted, _word_size);

    ASSERT_EQ(_mask.get_committed_size(),
        (size_t)0);
    ASSERT_EQ(_mask.get_committed_size_in_range(_base, _word_size),
        (size_t)0);

    // Now make an committed region
    size_t sr_word_size;
    const MetaWord* sr_base = calc_random_subrange(&sr_word_size);
    LOG("subrange " PTR_FORMAT "-" PTR_FORMAT ".",
        p2i(sr_base), p2i(sr_base + sr_word_size));

    ASSERT_EQ(_mask.get_committed_size_in_range(sr_base, sr_word_size),
              (size_t)0);
    for (const MetaWord* p = _base; p < _base + _word_size; p ++) {
      ASSERT_FALSE(_mask.is_committed_address(p));
    }

    size_t prior_committed = _mask.mark_range_as_committed(sr_base, sr_word_size);
    verify_mask();
    ASSERT_EQ(prior_committed, (size_t)0);

    // Again, for fun, should be a noop now.
    prior_committed = _mask.mark_range_as_committed(sr_base, sr_word_size);
    verify_mask();
    ASSERT_EQ(prior_committed, sr_word_size);

    ASSERT_EQ(_mask.get_committed_size_in_range(sr_base, sr_word_size),
        sr_word_size);
    ASSERT_EQ(_mask.get_committed_size(),
        sr_word_size);
    ASSERT_EQ(_mask.get_committed_size_in_range(_base, _word_size),
        sr_word_size);
    for (const MetaWord* p = _base; p < _base + _word_size; p ++) {
      if (p >= sr_base && p < sr_base + sr_word_size) {
        ASSERT_TRUE(_mask.is_committed_address(p));
      } else {
        ASSERT_FALSE(_mask.is_committed_address(p));
      }
    }

    // Re-uncommit whole range
    prior_uncommitted = _mask.mark_range_as_uncommitted(_base, _word_size);
    verify_mask();
    ASSERT_EQ(prior_uncommitted, _word_size - sr_word_size);

    EXPECT_EQ(_mask.get_committed_size_in_range(sr_base, sr_word_size),
        (size_t)0);
    EXPECT_EQ(_mask.get_committed_size(),
        (size_t)0);
    EXPECT_EQ(_mask.get_committed_size_in_range(_base, _word_size),
        (size_t)0);
    for (const MetaWord* p = _base; p < _base + _word_size; p ++) {
      ASSERT_FALSE(_mask.is_committed_address(p));
    }

  }


  void test3() {

    // arbitrary ranges are set and cleared and compared with the test map
    TestMap map(_word_size);

    _mask.clear_large();

    for (int run = 0; run < 100; run ++) {

      range_t r;
      calc_random_range(_word_size, &r, Settings::commit_granule_words());

      ASSERT_EQ(_mask.get_committed_size(), (size_t)map.get_num_set());
      ASSERT_EQ(_mask.get_committed_size_in_range(_base + r.from, r.to - r.from),
                (size_t)map.get_num_set(r.from, r.to));

      if (os::random() % 100 < 50) {
        _mask.mark_range_as_committed(_base + r.from, r.to - r.from);
        map.set_range(r.from, r.to);
      } else {
        _mask.mark_range_as_uncommitted(_base + r.from, r.to - r.from);
        map.clear_range(r.from, r.to);
      }

      ASSERT_EQ(_mask.get_committed_size(), (size_t)map.get_num_set());
      ASSERT_EQ(_mask.get_committed_size_in_range(_base + r.from, r.to - r.from),
                (size_t)map.get_num_set(r.from, r.to));

    }

  }


public:

  CommitMaskTest(const MetaWord* base, size_t size)
    : _base(base), _word_size(size), _mask(base, size)
  {}

  void test() {
    LOG("mask range: " PTR_FORMAT "-" PTR_FORMAT
         " (" SIZE_FORMAT " words).",
         p2i(_base), p2i(_base + _word_size), _word_size);
    for (int i = 0; i < 5; i ++) {
      test1(); test2(); test3();
    }
  }


};

TEST_VM(metaspace, commit_mask_basics) {

  const MetaWord* const base = (const MetaWord*) 0x100000;

  CommitMask mask1(base, Settings::commit_granule_words());
  ASSERT_EQ(mask1.size(), (BitMap::idx_t)1);

  CommitMask mask2(base, Settings::commit_granule_words() * 4);
  ASSERT_EQ(mask2.size(), (BitMap::idx_t)4);

  CommitMask mask3(base, Settings::commit_granule_words() * 43);
  ASSERT_EQ(mask3.size(), (BitMap::idx_t)43);

  mask3.mark_range_as_committed(base, Settings::commit_granule_words());
  mask3.mark_range_as_committed(base + (Settings::commit_granule_words() * 42), Settings::commit_granule_words());

  ASSERT_EQ(mask3.at(0), 1);
  for (int i = 1; i < 42; i ++) {
    ASSERT_EQ(mask3.at(i), 0);
  }
  ASSERT_EQ(mask3.at(42), 1);

}

TEST_VM(metaspace, commit_mask_small) {

  const MetaWord* const base = (const MetaWord*) 0x100000;

  CommitMaskTest test(base, Settings::commit_granule_words());
  test.test();

}

TEST_VM(metaspace, commit_mask_range) {

  const MetaWord* const base = (const MetaWord*) 0x100000;
  const size_t len = Settings::commit_granule_words() * 4;
  const MetaWord* const end = base + len;
  CommitMask mask(base, len);

  LOG("total range: " PTR_FORMAT "-" PTR_FORMAT "\n", p2i(base), p2i(end));

  size_t l = mask.mark_range_as_committed(base, len);
  ASSERT_LE(l, len);

  for (const MetaWord* p = base; p <= end - Settings::commit_granule_words();
       p += Settings::commit_granule_words()) {
    for (const MetaWord* p2 = p + Settings::commit_granule_words();
         p2 <= end; p2 += Settings::commit_granule_words()) {
      LOG(PTR_FORMAT "-" PTR_FORMAT "\n", p2i(p), p2i(p2));
      EXPECT_EQ(mask.get_committed_size_in_range(p, p2 - p),
                (size_t)(p2 - p));
    }
  }

  l = mask.mark_range_as_uncommitted(base, len);
  ASSERT_EQ(l, (size_t)0);

  for (const MetaWord* p = base; p <= end - Settings::commit_granule_words();
       p += Settings::commit_granule_words()) {
    for (const MetaWord* p2 = p + Settings::commit_granule_words();
         p2 <= end; p2 += Settings::commit_granule_words()) {
      LOG(PTR_FORMAT "-" PTR_FORMAT "\n", p2i(p), p2i(p2));
      EXPECT_EQ(mask.get_committed_size_in_range(p, p2 - p),
                (size_t)(0));
    }
  }

}


TEST_VM(metaspace, commit_mask_random) {

  for (int i = 0; i < 5; i ++) {

    // make up a range out of thin air
    const MetaWord* const base =
        align_down( (const MetaWord*) ((uintptr_t) os::random() * os::random()),
                    Settings::commit_granule_bytes());
    const size_t len = align_up( 1 + (os::random() % M),
                    Settings::commit_granule_words());

    CommitMaskTest test(base, len);
    test.test();

  }

}