summaryrefslogtreecommitdiff
path: root/tests/src/GC/Stress/Tests/pinstress.cs
blob: 30b4c58f929984c4564625932259a985a1d77f86 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.

// Licensed under the MIT license. See LICENSE file in the project root for full license information.
//

using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;

namespace PinStress
{
    internal class Node
    {
        private byte[] _b;
        private GCHandle _gch;
        private bool _freed = true;

        public Node()
        {
            _b = new byte[84900];
            _gch = GCHandle.Alloc(_b, GCHandleType.Pinned);
            _freed = false;
        }


        ~Node()
        {
            // in case someone forgets to call Free()
            Free();
        }


        public void Free()
        {
            // this check is necessary to avoid possibly double-freeing the handle
            if (!_freed)
            {
                _gch.Free();
                _freed = true;
                GC.SuppressFinalize(this);
            }
        }


        public bool IsFree
        {
            get
            {
                return _freed;
            }
        }
    }



    public class PinStress
    {
        public static int Main(string[] args)
        {
            Random r;
            int randomSeed = 0;
            float percentDelete = 0.0F;
            float percentFree = 0.0F;
            int numNodes = 0;
            int numPasses = 0;

            if (args.Length == 0)
            {
                // use defaults
                percentDelete = 0.6F;
                percentFree = 0.99F;
                numNodes = 10000;
                numPasses = 5;
            }
            else if (args.Length == 5)
            {
                //take command-line args for random seed, %delete, %free, numNodes, numPasses

                if (!Int32.TryParse(args[0], out randomSeed))
                {
                    Console.WriteLine("Invalid randomSeed");
                    return 0;
                }

                if (!Single.TryParse(args[1], System.Globalization.NumberStyles.Float, System.Globalization.CultureInfo.InvariantCulture, out percentDelete))
                {
                    Console.WriteLine("Invalid percentDelete");
                    return 0;
                }
                if ((percentDelete < 0) || (percentDelete > 1.0))
                {
                    Console.WriteLine("Invalid percentDelete");
                    return 0;
                }

                if (!Single.TryParse(args[2], System.Globalization.NumberStyles.Float, System.Globalization.CultureInfo.InvariantCulture, out percentFree))
                {
                    Console.WriteLine("Invalid percentFree");
                    return 0;
                }
                if ((percentFree < 0) || (percentFree > 1.0))
                {
                    Console.WriteLine("Invalid percentFree");
                    return 0;
                }

                if (!Int32.TryParse(args[3], out numNodes))
                {
                    Console.WriteLine("Invalid numNodes");
                    return 0;
                }
                if (numNodes <= 0)
                {
                    Console.WriteLine("Invalid numNodes");
                    return 0;
                }

                if (!Int32.TryParse(args[4], out numPasses))
                {
                    Console.WriteLine("Invalid numPasses");
                    return 0;
                }
            }
            else
            {
                Console.WriteLine("USAGE: pinstress.exe <randomSeed> <percentDelete> <percentFree> <numNodes> <numPasses>");
                Console.WriteLine("\t where percent* is a float between 0.0 and 1.0");
                return 0;
            }

            r = new Random(randomSeed);
            List<Node> list = new List<Node>();

            // loop
            for (int j = 0; j != numPasses; j++)
            {
                // repopulate the list
                while (list.Count < numNodes)
                {
                    list.Add(new Node());
                }

                // unpin percentFree% of the nodes
                for (int i = 0; i < numNodes * percentFree; i++)
                {
                    int index = r.Next(list.Count - 1);
                    if (!list[index].IsFree)
                    {
                        // only unpin if we haven't already freed
                        list[index].Free();
                    }
                }

                // delete percentDelete% of the freed nodes
                for (int i = 0; i < numNodes * percentDelete; i++)
                {
                    int index = r.Next(list.Count - 1);
                    if (list[index].IsFree)
                    {
                        // this will fragment the heap
                        list.RemoveAt(index);
                    }
                }

                // shrink the List to free up as much memory as possible
                list.TrimExcess();

                Console.WriteLine("Pass {0}", j);
            }

            Console.WriteLine("Test Passed");
            return 100;
        }
    }
}