/*
 * Copyright (c) 2011, 2020, 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.
 */

/*
 * @test
 * @key stress randomness
 *
 * @summary converted from VM Testbase gc/gctests/OneeFinalizerTest.
 * VM Testbase keywords: [gc, stress, stressopt, nonconcurrent, jrockit]
 * VM Testbase readme:
 * DESCRIPTION
 * Regression test that verifies that only one finalizer gets called.
 *
 * COMMENTS
 * This test was ported from JRockit test suite.
 *
 * @library /vmTestbase
 *          /test/lib
 * @run main/othervm
 *      -XX:-UseGCOverheadLimit
 *      -Xlog:gc:gc.log
 *      gc.gctests.OneeFinalizerTest.OneeFinalizerTest
 */

package gc.gctests.OneeFinalizerTest;

import nsk.share.TestFailure;
import nsk.share.gc.GC;
import nsk.share.gc.GCTestBase;
import nsk.share.gc.gp.GarbageUtils;
import nsk.share.test.Stresser;

/**
 * Test that verifies that finalize() method is invoking only once.
 */
public class OneeFinalizerTest extends GCTestBase {

    private GlobalSafeCounter[] finalizerCounters = null;

    /**
     * Helper class used for counting number of calls to finalizers.
     */
    protected class GlobalSafeCounter {

        private int counter;

        /**
         * Constructor that inits the global counter to 0.
         */
        protected GlobalSafeCounter() {
            counter = 0;
        }

        /**
         * Reset the global counter to 0.
         */
        protected final void resetCounter() {
            synchronized (this) {
                counter = 0;
            }
        }

        /**
         * Increase the global counter by 1.
         */
        protected final void increaseCounter() {
            synchronized (this) {
                counter++;
            }
        }

        /**
         * Retrieve the global counter value.
         *
         * @return value of the global counter
         */
        protected final int getCounterValue() {
            int value;

            synchronized (this) {
                value = counter;
            }

            return value;
        }
    }

    /**
     * Helper class the implements finalize(), and that increments
     * the global counters for each finalize() invokation.
     */
    protected class FinalizedObject {

        private final int counterIndex;

        /**
         * Constructor for the helper object which implements finalize().
         *
         * @param index Index for the counter in the global array, that
         * corresponds to this object.
         */
        protected FinalizedObject(int index) {
            counterIndex = index;
        }

        /**
         * Increases the global counter for this object when finalize()
         * gets called (to make sure each finalizer gets called onee).
         */
        @Override
        protected final void finalize() {
            finalizerCounters[counterIndex].increaseCounter();
        }
    }

    private void initOneeFinalizerTest(int numberOfObjects) {
        // NOTE: Set to null in case it's been used before (to prevent OOM)
        finalizerCounters = null;
        finalizerCounters = new GlobalSafeCounter[numberOfObjects];

        for (int i = 0; i < numberOfObjects; i++) {
            finalizerCounters[i] = new GlobalSafeCounter();
        }
    }

    /**
     * Tests that the finalize() method on each FinalizedObject instance
     * has been called exactly one time.
     */
    @Override
    public void run() {


        int numberOfObjects = 2000;

        initOneeFinalizerTest(numberOfObjects);

        FinalizedObject[] testObjects = new FinalizedObject[numberOfObjects];

        // creates garbage
        for (int i = 0; i < numberOfObjects; i++) {
            testObjects[i] = new FinalizedObject(i);
        }

        if (testObjects[0].hashCode() == 212_85_06) {
            System.out.println("Bingo!!!");
        }

        testObjects = null;

        Stresser stresser = new Stresser(runParams.getStressOptions());
        stresser.start(0);
        /* force finalization  */
        GarbageUtils.eatMemory(stresser);
        if (!stresser.continueExecution()) {
            // may be we didn't eat all memory and didn't provoke GC
            System.out.println("Passed without check");
            return;
        }
        System.gc();
        System.runFinalization();
        System.gc();
        System.runFinalization();
        System.gc();

        int numberOfFinalizersRunMoreThanOnce = 0;
        int numberOfFinalizersNotRun = 0;

        for (int i = 0; i < numberOfObjects; i++) {
            int counter = finalizerCounters[i].getCounterValue();
            if (counter > 1) {
                numberOfFinalizersRunMoreThanOnce++;
                System.err.println("Object #" + i + " counter = " + counter);
            } else if (counter == 0) {
                System.err.println("WARNING: Finalizer not run for object #" + i);
                numberOfFinalizersNotRun++;
            }
        }

        if (numberOfFinalizersNotRun > 0) {
            System.err.println("WARNING: " + numberOfFinalizersNotRun + " finalizers not run");
        }

        if (numberOfFinalizersRunMoreThanOnce != 0) {
            throw new TestFailure("OneeFinalizerTest failed. " + numberOfFinalizersRunMoreThanOnce + " errors");
        }
        System.out.println("Test passed.");
    }

    public static void main(String[] args) {
        GC.runTest(new OneeFinalizerTest(), args);
    }
}