All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jgroups.util.RetransmitTable Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Final
Show newest version
package org.jgroups.util;

import org.jgroups.Message;
import org.jgroups.logging.Log;
import org.jgroups.logging.LogFactory;

import java.util.LinkedList;
import java.util.List;

/**
 * A store for messages to be retransmitted or delivered. Used on sender and receiver side, as a replacement for
 * HashMap. RetransmitTable should use less memory than HashMap, as HashMap.Entry has 4 fields, plus arrays for storage.
 * 

* RetransmitTable maintains a matrix (an array of arrays) of messages. Messages are stored in the matrix by mapping * their seqno to an index. E.g. when we have 10 rows of 1000 messages each, and first_seqno is 3000, then a message with * seqno=5600, will be stored in the 3rd row, at index 600. *

* Rows are removed when all messages in that row have been received.

* This class in not synchronized; the caller has to make sure access to it is synchronized * @author Bela Ban */ public class RetransmitTable { protected final int num_rows; /** Must be a power of 2 for efficient modular arithmetic **/ protected final int msgs_per_row; protected final double resize_factor; protected Message[][] matrix; /** The first seqno, at matrix[0][0] */ protected long offset; protected int size=0; /** The highest seqno purged */ protected long highest_seqno_purged; /** The highest seqno in the table */ protected long highest_seqno; /** Time (in ms) after which a compaction should take place. 0 disables compaction */ protected long max_compaction_time=DEFAULT_MAX_COMPACTION_TIME; /** The time when the last compaction took place. If a {@link #compact()} takes place and sees that the * last compaction is more than max_compaction_time ms ago, a compaction will take place */ protected long last_compaction_timestamp=0; /** By default, rows are only nulled and highest_seqno_purged is adjusted when {@link #purge(long)} is called. * When automatic_purging is enabled (default is off), rows are purged and highest_seqno_purged is adjusted * on {@link #remove(long)} */ protected boolean automatic_purging; protected static final long DEFAULT_MAX_COMPACTION_TIME=2 * 60 * 1000L; protected static final double DEFAULT_RESIZE_FACTOR=1.2; protected static final Log log=LogFactory.getLog(RetransmitTable.class); public RetransmitTable() { this(5, 8192, 0, DEFAULT_RESIZE_FACTOR); } public RetransmitTable(int num_rows, int msgs_per_row, long offset) { this(num_rows, msgs_per_row, offset, DEFAULT_RESIZE_FACTOR); } public RetransmitTable(int num_rows, int msgs_per_row, long offset, double resize_factor) { this(num_rows, msgs_per_row, offset, resize_factor, DEFAULT_MAX_COMPACTION_TIME, false); } public RetransmitTable(int num_rows, int msgs_per_row, long offset, double resize_factor, long max_compaction_time, boolean automatic_purging) { this.num_rows=num_rows; this.msgs_per_row=Util.getNextHigherPowerOfTwo(msgs_per_row); this.resize_factor=resize_factor; this.max_compaction_time=max_compaction_time; this.automatic_purging=automatic_purging; this.offset=this.highest_seqno_purged=this.highest_seqno=offset; matrix=new Message[num_rows][]; if(resize_factor <= 1) throw new IllegalArgumentException("resize_factor needs to be > 1"); } public int getLength() { return matrix.length; } public long getOffset() { return offset; } /** Returns the total capacity in the matrix */ public int capacity() {return matrix.length * msgs_per_row;} /** Returns the numbers of messages in the table */ public int size() {return size;} public boolean isEmpty() {return size <= 0;} public long getHighest() { return highest_seqno; } public long getHighestPurged() { return highest_seqno_purged; } public long getMaxCompactionTime() { return max_compaction_time; } public void setMaxCompactionTime(long max_compaction_time) { this.max_compaction_time=max_compaction_time; } public boolean isAutomaticPurging() { return automatic_purging; } public void setAutomaticPurging(boolean automatic_purging) { this.automatic_purging=automatic_purging; } /** Returns the ratio between size and capacity, as a percentage */ public double getFillFactor() { return size == 0? 0.0 : (int)(((double)size / capacity()) * 100); } /** * Adds a new message to the index computed as a function of seqno * @param seqno * @param msg * @return True if the element at the computed index was null, else false */ public boolean put(long seqno, Message msg) { return putIfAbsent(seqno, msg) == null; } /** * Adds a message if the element at the given index is null. Returns null if no message existed at the given index, * else returns the existing message and doesn't set the element. * @param seqno * @param msg * @return The existing message, or null if there wasn't any */ public Message putIfAbsent(long seqno, Message msg) { int row_index=computeRow(seqno); if(row_index >= matrix.length) { resize(seqno); row_index=computeRow(seqno); } Message[] row=getRow(row_index); int index=computeIndex(seqno); Message existing_msg=row[index]; if(existing_msg == null) { row[index]=msg; size++; if(seqno > highest_seqno) highest_seqno=seqno; return null; } else return existing_msg; } public Message get(long seqno) { int row_index=computeRow(seqno); if(row_index < 0 || row_index >= matrix.length) return null; Message[] row=matrix[row_index]; if(row == null) return null; int index=computeIndex(seqno); return index >= 0? row[index] : null; } public List get(long from, long to) { List retval=null; for(long seqno=from; seqno <= to; seqno++) { Message msg=get(seqno); if(msg != null) { if(retval == null) retval=new LinkedList<>(); retval.add(msg); } } return retval; } /** Removes the message with seqno from the table, nulls the index */ public Message remove(long seqno) { int row_index=computeRow(seqno); if(row_index < 0 || row_index >= matrix.length) return null; Message[] row=matrix[row_index]; if(row == null) return null; int index=computeIndex(seqno); if(index < 0) return null; Message existing_msg=row[index]; if(existing_msg != null) { row[index]=null; size=Math.max(size-1, 0); // cannot be < 0 (well that would be a bug, but let's have this 2nd line of defense !) if(automatic_purging) { if(seqno > highest_seqno_purged) highest_seqno_purged=seqno; } } return existing_msg; } /** Removes all elements. This method is usually called just before removing a retransmit table, so typically * it is not used anymore after returning */ public void clear() { matrix=new Message[num_rows][]; size=0; offset=highest_seqno_purged=highest_seqno=0; } /** * Removes all messages less than or equal to seqno from the table. Does this by nulling entire rows in the matrix * and nulling all elements < index(seqno) of the first row that cannot be removed * @param seqno */ public void purge(long seqno) { int num_rows_to_remove=(int)(seqno - offset) / msgs_per_row; for(int i=0; i < num_rows_to_remove; i++) // Null all rows which can be fully removed matrix[i]=null; int row_index=computeRow(seqno); if(row_index < 0 || row_index >= matrix.length) return; Message[] row=matrix[row_index]; if(row != null) { int index=computeIndex(seqno); for(int i=0; i <= index; i++) // null all messages up to and including seqno in the given row row[i]=null; } size=computeSize(); if(seqno > highest_seqno_purged) highest_seqno_purged=seqno; // see if compaction should be triggered if(max_compaction_time <= 0) return; long current_time=System.currentTimeMillis(); if(last_compaction_timestamp > 0) { if(current_time - last_compaction_timestamp >= max_compaction_time) { compact(); last_compaction_timestamp=current_time; } } else last_compaction_timestamp=current_time; } /** Moves rows down the matrix, by removing purged rows. If resizing to accommodate seqno is still needed, computes * a new size. Then either moves existing rows down, or copies them into a new array (if resizing took place) */ protected void resize(long seqno) { int num_rows_to_purge=(int)((highest_seqno_purged - offset) / msgs_per_row); int row_index=computeRow(seqno) - num_rows_to_purge; if(row_index < 0) return; int new_size=Math.max(row_index +1, matrix.length); if(new_size > matrix.length) { Message[][] new_matrix=new Message[new_size][]; System.arraycopy(matrix, num_rows_to_purge, new_matrix, 0, matrix.length - num_rows_to_purge); matrix=new_matrix; } else if(num_rows_to_purge > 0) { move(num_rows_to_purge); } offset+=(num_rows_to_purge * msgs_per_row); size=computeSize(); } /** Moves contents of matrix num_rows down. Avoids a System.arraycopy() */ protected void move(int num_rows) { if(num_rows <= 0 || num_rows > matrix.length) return; int target_index=0; for(int i=num_rows; i < matrix.length; i++) matrix[target_index++]=matrix[i]; for(int i=matrix.length - num_rows; i < matrix.length; i++) matrix[i]=null; } /** * Moves the contents of matrix down by the number of purged rows and resizes the matrix accordingly. The * capacity of the matrix should be size * resize_factor */ public void compact() { // This is the range we need to copy into the new matrix (including from and to) int from=computeRow(highest_seqno_purged), to=computeRow(highest_seqno); int range=to - from +1; // e.g. from=3, to=5, new_size has to be [3 .. 5] (=3) int new_size=(int)Math.max(range * resize_factor, range +1); new_size=Math.max(new_size, num_rows); // don't fall below the initial size defined if(new_size < matrix.length) { if(log.isTraceEnabled()) log.trace("compacting matrix from " + matrix.length + " rows to " + new_size + " rows"); Message[][] new_matrix=new Message[new_size][]; System.arraycopy(matrix, from, new_matrix, 0, range); matrix=new_matrix; offset+=from * msgs_per_row; size=computeSize(); } } /** Iterate from highest_seqno_purged to highest_seqno and add up non-null values */ public int computeSize() { int retval=0; int from=computeRow(highest_seqno_purged), to=computeRow(highest_seqno); for(int i=from; i <= to; i++) { Message[] row=matrix[i]; if(row == null) continue; for(int j=0; j < row.length; j++) { if(row[j] != null) retval++; } } return retval; } /** Returns the number of null elements in the range [from .. to], excluding 'from' and 'to' */ public int getNullMessages(long from, long to) { int retval=0; for(long i=from+1; i < to; i++) { int row_index=computeRow(i); if(row_index < 0 || row_index >= matrix.length) continue; Message[] row=matrix[row_index]; if(row == null || row[computeIndex(i)] == null) retval++; } return retval; } public String toString() { StringBuilder sb=new StringBuilder(); sb.append("size=" + size + ", capacity=" + capacity() + ", highest=" + highest_seqno + ", highest_purged=" + highest_seqno_purged); return sb.toString(); } /** Dumps the seqnos in the table as a list */ public String dump() { StringBuilder sb=new StringBuilder(); boolean first=true; for(int i=0; i < matrix.length; i++) { Message[] row=matrix[i]; if(row == null) continue; for(int j=0; j < row.length; j++) { if(row[j] != null) { long seqno=offset + (i * msgs_per_row) + j; if(first) first=false; else sb.append(", "); sb.append(seqno); } } } return sb.toString(); } /** Dumps the non-null in the table in a pseudo graphic way */ public String dumpMatrix() { StringBuilder sb=new StringBuilder(); for(int i=0; i < matrix.length; i++) { Message[] row=matrix[i]; sb.append(i + ": "); if(row == null) { sb.append("\n"); continue; } for(int j=0; j < row.length; j++) { if(row[j] != null) sb.append("* "); else sb.append(" "); } sb.append("\n"); } return sb.toString(); } /** * Computes the size of all messages currently in the table. This includes messages that haven't been purged or * compacted yet. * @param include_headers If true, {@link org.jgroups.Message#size()} is used, which will include the size of all * headers and the dest and src addresses. Else {@link org.jgroups.Message#getLength()} is used to compute. * Note that the latter is way more efficient. * @return Number of bytes of all messages. */ public long sizeOfAllMessages(boolean include_headers) { long retval=0; for(int i=0; i < matrix.length; i++) { Message[] row=matrix[i]; if(row == null) continue; for(int j=0; j < row.length; j++) { Message msg=row[j]; if(msg != null) retval+=include_headers? msg.size() : msg.getLength(); } } return retval; } /** * Returns a row. Creates a new row and inserts it at index if the row at index doesn't exist * @param index * @return A row */ protected Message[] getRow(int index) { Message[] row=matrix[index]; if(row == null) { row=new Message[msgs_per_row]; matrix[index]=row; } return row; } /** Computes and returns the row index for seqno */ protected int computeRow(long seqno) { int diff=(int)(seqno-offset); if(diff < 0) return diff; return diff / msgs_per_row; } /** Computes and returns the index within a row for seqno */ protected int computeIndex(long seqno) { int diff=(int)(seqno - offset); if(diff < 0) return diff; // Same as diff % msgs_per_row return diff & (msgs_per_row - 1); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy