Cegal tech blog

JVM memory leak

Written by Cegal Tech Advisors | Jul 6, 2015 5:53:00 PM

Using Oracle Cloud Control 12C to Analyse a Memory Leak Problem.

1. Introduction

This post aims to use the Oracle Cloud Control 12C to find the root cause of a memory leak problem. One of the main advantages of Java is the use of a garbage collector that allows programmers to forget about memory management. That is not the case in languages such as C++ where you have to allocate and free memory for your code. In my experience in a big company, I saw memory leak problems several times, causing the unavailability or the bad performance of many applications.

In addition, a memory leak belongs to a class of situations called software aging problems. Software aging defines the loss of performance over time because of the gradual accumulation of little problems. Other term within this kind of problem is the rejuvenation process. For example, rejuvenation happens when the system has to be restarted in order to free the memory accumulated by a Java Virtual Machine (JVM) process after a period of time (COTRONEO, D, et al., 2015).

However, there is a question about this problem on JVM. Taking into account that JVM has a garbage collector, how does JVM accumulate memory over time? The answer is in relationship to the way that the garbage collector uses to free memory, the codification on Java and even the use of some parameters within the configuration.

JVM uses a generational model to manage the memory assigned to itself so the following lines will explain the dynamic of the JVM memory. After this, Oracle Cloud Control 12 C will be used to diagnose the cause of the problem.

2. Java Memory

The Java memory model used here is the Oracle Hot Spot model, that is composed of the sectors that can be seen in the following picture.

Following each one of these components will be described.

2.1. Heap

This is the memory place where objects (instances of a class) are saved. With this in mind, taking into account that Java is a garbage collector language, the minimum and maximum thresholds have to be tuned carefully in order to optimize the collection of objects without any reference. It means objects that are not used by any object (Oracle, n.d.). In addition, we have to analyze the sizing of our system to avoid the CPU consumption caused by garbage collection execution and consequently crash of the system due to the lack of enough memory. How can I set the size of the heap? You have to use these parameters:

  • Heap minimum size: -Xms
  • Heap maximum size: -Xmx

Furthermore, the heap is divided into two places: the young generation and the old generation.

2.1.1. Young generation

This is the place where objects generated after the execution of the sentence new() are saved. This memory space is divided in two part the new generation or Eden and the survivor space (Oracle, n.d.).

2.1.1.1. Eden or new generation

When an object has just been created using the sentence new(), it is saved here. When this place is full, a minor garbage collection is executed to eliminate objects without any reference. Referenced objects are moved to one of the survivor spaces. In addition, the age of referenced objects is incremented by one (Oracle, n.d.).

2.1.1.2. Survivor space

This place keeps referenced objects, which are moved from the Eden during the minor collection execution. There are two survivor spaces S0 and S1 (Oracle, n.d.).

2.1.2. Old generation

This segment of memory contains older objects that are still referenced. This objects are collected from the young generation when the execution of a minor garbage collection found objects that have passed the age threshold. This threshold is defined by the parameter: MaxTenuringThreshold (Oracle, n.d.).

2.1.3. A little description of the garbage collector work on the heap.

Suppose we have a JVM whose MaxTenuringThreshold was set on 4 and a class called RM.java, which will be instanced several times (Oracle, n.d.).

i. After creating several objects from the class RM. Java, using the instruction RM rm1=new RM(), will be stored in the Eden within the young generation space (Oracle, n.d.).

ii. After some time, the Eden will be filled by the creation of objects, and then a minor garbage collection will be triggered. This minor collection will eliminate objects without reference and will move the objects which are being used (they have references) into the survivor segment, let us say S0 (Oracle, n.d.).

iii. If the Eden space is filled again because of the use, the minor collection will be executed. In this case, the survivor objects (with references in use) will be moved into the survivor space called S1. In addition, objects with references from the survivor space S0 will be moved into the S0 segment, increasing their age by 1. Eventually, all the survivor objects are moved to the S1 and both the Eden and the S1 will be empty (Oracle, n.d.).

iv. When a minor collection is executed again, the previous process runs again, but this time, survivor objects are moved from the Eden into the S0 and from the S1 into the S0. In addition, the age of survivor objects is increased by 1 (Oracle, n.d.).

v. Over time, some will reach the age of 4 (MaxTenuringThreshold=4), and then these objects will be moved into the old generation space (Oracle, n.d.).

vi. When the old generation space is full, a major garbage collection is executed over this segment (Oracle, n.d.).

2.2. Non heap space

This space contains the JVM native code and the classes’ metadata. These classes are loaded by the applications deployed on the server. It is important to remark that this segment stores metadata. It does not store objects (Oracle, n.d.). For example, native code used to improve the Just in Time compiler uses this space. In this guide, the memory leaks of this space will not be analyzed.

3. Memory leak analysis

Now we have some concepts about Java memory, we can analyze a memory leak problem using Oracle Cloud Control 12c.

3.1. Configuring the Java Pool

From the setup menu, select Middleware Management > Application Performance Management option

Select the JVMD engine and click on Configure.

In the JVMs and Pools tab, select the pool related to the managed server that contains our application and click on Configure.

In this screen, modify the path where the heap dump will be saved and click on Save.

3.2. Using Grinder to stress the application.

In this case, the application has a bug that will generate a memory leak scenario. To accelerate the process, Grinder is used to create thousands of requests.

This is the heap state before executing the test

It is important to realize that before the test, the number of major garbage collections is zero.

I generated a heap dump report before stressing the application server. These are the steps.

  1. Select the option “Heap Snapshots and Class Histograms”

  1. Click on “Create.”

  1. Select “JVMD Format (txt)” and “All.”

  1. Click on “Submit.”

  1. This is the result of the job, which executes the heap dump.

  1. In this table, it is possible to see the new heap loaded within the system

Execution of Grinder, now it is time to stress the application using Grinder to simulate several sessions.

Start the console

Start the agent

Start the application requests

After some minutes the pressure can be felt in the number of Major GC that now equals 9.

Another interesting indicator is that even though 9 major garbage collections were executed, the “JVM heap used after GC” is increasing.

3.3. Getting the memory leak report

In order to get the memory leak report to find the root cause of this problem, you have the follow these steps.

  1. Right-click on the server affected and then select Diagnostics > Heap Snapshots and Class Histograms

  1. Click on Create

  1. Select the radio button “JVMD Format (txt)” and the check box “All.”

  1. Click on Submit

  1. After some minutes, the job in charge of the heap processing will finish showing something like this.

  1. In order to get the report, you have to select one of the heap generated by the previous process and then click on Detail.

  1. Click on the tab Memory Leak Report

  1. According to this page, we have a memory leak problem and two candidates

  1. So we have to get into details to see who is causing the problem

In the first case, we have the class weblogic.servlet.internal.HttpServer.SessionLogin

In the second case, we have the class weblogic.servlet.internal.session.MemorySessionContext

With this in mind, it seems that the problem is related to the management of the application’s web sessions. Thus, we have to review the web.xml file to get some details about this. This is the file.

 
After reviewing the file, we can see there is a problem with the timeout. For some reason it was set in “-1”, meaning the web session would live forever, and then the garbage collector could not free the JVM heap. This case happened in a big company, but the difference was that there were several web applications. However, I think that this guide could be useful to learn about how to find the root cause of a memory leak problem using the Oracle Cloud Control, at that time I had to use the Eclipse Memory Analyser Tool, but it is another history.
 
Last but not least, it is advisable to use the methodologies to find the root cause of a problem to avoid scheduled restarts to rejuvenate the application server.