mq5.1-source.src.share.cclient.io.SocketTest.cpp Maven / Gradle / Ivy
The newest version!
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2000-2013 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
/*
* @(#)SocketTest.cpp 1.3 06/26/07
*/
#include
#include "../util/UtilityMacros.h"
#include "../util/PRTypesUtils.h"
#include "../basictypes/Monitor.hpp"
#include "TCPSocket.hpp"
//#define DUMP_TCP_STREAM
#if defined(DUMP_TCP_STREAM)
# if defined(WIN32)
static TCPSocket sock("c:/temp/socket");
# else
static TCPSocket sock("/home/de134463/temp/socket");
# endif
#else
static TCPSocket sock;
#endif
static TCPSocket serverSocket;
static iMQError socketError;
static Monitor monitor;
#if !defined(WIN32)
extern "C" {
#endif // !defined(WIN32)
static void socketReader(void *arg);
static void socketWriter(void *arg);
static void socketServer(void *arg);
static void socketReaderWriter(void *arg);
#if !defined(WIN32)
}
#endif // !defined(WIN32)
static int MAX_SLEEP_MS_WRITER = 0;
static int MAX_SLEEP_MS_READER = 0;
static int MAX_ITER = 0;
static int PRINT_INC = 0;
static const PRUint16 SERVER_PORT = 22500;
static const PRUint8 WRITE_BYTE = 10;
static const PRUint8 READ_BYTE = 40;
static const PRUint32 TEST_DIV_VALUE = 1000;
static const PRUint32 TEST_MOD_VALUE = 256;
static const int MAX_BYTES = 1000000;
static PRInt32 g_bytesPerIter = 0;
static PRInt32 g_numIters = 0;
static PRInt32 g_printInc = 0;
static PRInt32 g_timeoutMSWriter = 0;
static PRInt32 g_timeoutMSReader = 0;
static PRInt32 g_timeoutMSServer = 0;
static PRInt32 g_maxSleepMSWriter = 0;
static PRInt32 g_maxSleepMSReader = 0;
static PRInt32 g_maxSleepMSServer = 0;
static float g_writerCloseProb = 0.0;
static float g_readerCloseProb = 0.0;
static float g_serverCloseProb = 0.0;
bool initializeGlobals(FILE * const inputFile)
{
if (inputFile == NULL) {
return false;
}
PRInt32 itemsRead = 0;
itemsRead += fscanf(inputFile, "%d,", &g_bytesPerIter);
itemsRead += fscanf(inputFile, "%d,", &g_numIters);
itemsRead += fscanf(inputFile, "%d,", &g_printInc);
itemsRead += fscanf(inputFile, "%d,", &g_timeoutMSWriter);
itemsRead += fscanf(inputFile, "%d,", &g_timeoutMSReader);
itemsRead += fscanf(inputFile, "%d,", &g_timeoutMSServer);
itemsRead += fscanf(inputFile, "%d,", &g_maxSleepMSWriter);
itemsRead += fscanf(inputFile, "%d,", &g_maxSleepMSReader);
itemsRead += fscanf(inputFile, "%d,", &g_maxSleepMSServer);
itemsRead += fscanf(inputFile, "%f,", &g_writerCloseProb);
itemsRead += fscanf(inputFile, "%f,", &g_readerCloseProb);
itemsRead += fscanf(inputFile, "%f\n", &g_serverCloseProb);
fprintf(stderr, "%d %d %d %d %d %d %d %d %d %f %f %f\n",
g_bytesPerIter, g_numIters, g_printInc, g_timeoutMSWriter,
g_timeoutMSReader, g_timeoutMSServer, g_maxSleepMSWriter,
g_maxSleepMSReader, g_maxSleepMSServer, g_writerCloseProb,
g_readerCloseProb, g_serverCloseProb);
return itemsRead == 12;
}
// Define this to use an echo server on the local machine
#define LOCAL_ECHO_SERVER
// Define this to only be an echo server
//#define ONLY_ECHO_SERVER
// Define this to have separate reader and writer threads
#define SEPARATE_READER_WRITER
static bool g_serverListen;
iMQError
socketTest(FILE * const inputFile)
{
//
ASSERT( inputFile != NULL );
if (inputFile == NULL) {
return IMQ_NULL_PTR_ARG;
}
// Skip the first line of the input file
char firstLine[1000];
fgets( firstLine, sizeof(firstLine), inputFile);
g_serverListen = true;
// Create the echo server
#if defined(LOCAL_ECHO_SERVER)
PRThread * serverThread = PR_CreateThread(PR_SYSTEM_THREAD,
socketServer,
NULL,
PR_PRIORITY_NORMAL,
PR_GLOBAL_THREAD,
PR_JOINABLE_THREAD,
0);
// sleep for a little bit, to let the server start up
PR_Sleep(PR_MicrosecondsToInterval(1 * 1000 * 1000));
# if defined(ONLY_ECHO_SERVER)
PR_JoinThread(serverThread);
return IMQ_SUCCESS;
# endif // defined(ONLY_ECHO_SERVER)
#endif // defined(LOCAL_ECHO_SERVER)
for (int iter = 0; initializeGlobals(inputFile); iter++) {
fprintf(stderr, "On iteration %d\n", iter);
#if defined(LOCAL_ECHO_SERVER)
RETURN_IF_ERROR( sock.connect("localhost", SERVER_PORT, PR_FALSE, 0xFFFFFFFF) );
#else
RETURN_IF_ERROR( sock.connect("129.153.130.220", 22228, PR_FALSE, 0xFFFFFFFF) );
//RETURN_IF_ERROR( sock.connect("129.153.138.193", SERVER_PORT, 0xFFFFFFFF) );
#endif
socketError = IMQ_SUCCESS;
#if defined(SEPARATE_READER_WRITER)
PRThread * readerThread = PR_CreateThread(PR_SYSTEM_THREAD,
socketReader,
NULL,
PR_PRIORITY_NORMAL,
PR_GLOBAL_THREAD,
PR_JOINABLE_THREAD,
0);
PRThread * writerThread = PR_CreateThread(PR_SYSTEM_THREAD,
socketWriter,
NULL,
PR_PRIORITY_NORMAL,
PR_GLOBAL_THREAD,
PR_JOINABLE_THREAD,
0);
PR_JoinThread(readerThread);
PR_JoinThread(writerThread);
#else
PRThread * readerWriterThread = PR_CreateThread(PR_SYSTEM_THREAD,
socketReaderWriter,
NULL,
PR_PRIORITY_NORMAL,
PR_GLOBAL_THREAD,
PR_JOINABLE_THREAD,
0);
PR_JoinThread(readerWriterThread);
#endif
//fprintf(stderr, "Sleeping for .3 secs. ");
PR_Sleep(PR_MicrosecondsToInterval(300 * 1000));
//fprintf(stderr, "Done. \n");
sock.close();
}
fclose(inputFile);
g_serverListen = false;
#if defined(LOCAL_ECHO_SERVER)
PR_Interrupt(serverThread);
PR_JoinThread(serverThread);
#endif // defined(LOCAL_ECHO_SERVER)
return IMQ_SUCCESS;
}
/**
* Echo server that listens on port SERVER_PORT.
*/
static void socketServer(void *arg)
{
UNUSED(arg);
iMQError errorCode = IMQ_SUCCESS;
TCPSocket * acceptSocket = NULL;
PRBool doEcho = PR_TRUE;
fprintf(stderr, " Binding to %d.", (PRInt32)SERVER_PORT);
//ERRCHK( serverSocket.bind("localhost", SERVER_PORT) );
ERRCHK( serverSocket.bind("129.153.138.193", SERVER_PORT) );
fprintf(stderr, " Bound.");
while (g_serverListen) {
ERRCHK( serverSocket.listen() != IMQ_SUCCESS );
ERRCHK( serverSocket.accept(&acceptSocket, 0xFFFFFFFF) != IMQ_SUCCESS );
// Echo until the connection goes away.
PRInt32 totalBytesRead = 0;
PRInt32 totalBytesWritten = 0;
while (doEcho) {
const PRInt32 MAX_BYTES_TO_READ = 1000000;
PRInt32 numBytesRead, numBytesWritten;
static PRUint8 bytesRead[MAX_BYTES_TO_READ];
if ((errorCode = acceptSocket->read(MAX_BYTES_TO_READ,
g_timeoutMSServer,
bytesRead,
&numBytesRead)) != IMQ_SUCCESS)
{
break;
}
if (numBytesRead > 0) {
fprintf(stderr, "%d total bytes read\n", totalBytesRead + numBytesRead );
}
int i;
for (i = 0; i < numBytesRead; i++) {
if (bytesRead[i] != (((totalBytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE)) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketReader, bytes[%d]=bytes[%d]=%d, expected %d, bytesRead=%d, numBytesRead=%d",
totalBytesRead + i, i, bytesRead[i], ((totalBytesRead + i) % TEST_MOD_VALUE),
totalBytesRead, numBytesRead ));
}
}
for (i = 0; i < numBytesRead; i++) {
ASSERT( bytesRead[i] == (((totalBytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE) );
}
if ((errorCode = acceptSocket->write(numBytesRead,
bytesRead,
0xFFFFFFFF,
&numBytesWritten)) != IMQ_SUCCESS)
{
break;
}
if (numBytesWritten > 0) {
fprintf(stderr, " %d total bytes written\n", totalBytesWritten + numBytesWritten );
}
totalBytesRead += numBytesRead;
totalBytesWritten += numBytesWritten;
if (numBytesRead > 0) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketServer, totalBytesRead=%d, totalBytesWritten=%d",
totalBytesRead, totalBytesWritten ));
}
if (numBytesRead != numBytesWritten) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketServer, bytesRead=%d, bytesWritten=%d, errorCode=%d",
numBytesRead, numBytesWritten, errorCode ));
}
ASSERT( numBytesRead == numBytesWritten );
// randomly close the socket
double prob = ((double)rand()/(double)RAND_MAX);
/*
static int cnt = 0;
cnt++;
if (prob < .001) {
fprintf( stderr, "Prob = %f, %d since (close on = %f)\n", prob, cnt, g_serverCloseProb);
cnt = 0;
}*/
if (prob < g_serverCloseProb) {
fprintf(stderr, "Server closing the socket early\n");
//acceptSocket->close();
break;
}
}
fprintf(stderr, "Server closing the socket\n");
acceptSocket->close();
DELETE( acceptSocket );
}
Cleanup:
fprintf(stderr, "\nsocketServer failed because %d.\n", errorCode);
DELETE( acceptSocket );
serverSocket.close();
return;
}
static void socketReaderWriter(void *arg)
{
UNUSED(arg);
static PRUint8 bytes[MAX_BYTES];
PRInt32 numBytesRead = 0;
PRInt32 numBytesWritten = 0;
PRInt32 error = IMQ_SUCCESS;
PRInt32 totalBytesToRead = g_bytesPerIter * g_numIters;
PRInt32 totalBytesToWrite = g_bytesPerIter * g_numIters;
int bytesRead = 0;
int bytesWritten = 0;
while ((bytesRead < totalBytesToRead) && (bytesWritten < totalBytesToWrite))
{
//
// Reader block
//
{
numBytesRead = 0;
// Initialize bytes
int i;
for (i = 0; i < g_bytesPerIter; i++) {
bytes[i] = READ_BYTE;
}
//error = sock.read(g_bytesPerIter, g_timeoutMSReader, bytes, &numBytesRead);
numBytesRead = PR_Recv(sock.hostSocket, bytes, g_bytesPerIter, 0, 0);
error = IMQ_SUCCESS;
if (numBytesRead < 0) {
numBytesRead = 0;
}
if (numBytesRead > 0) {
fprintf(stderr, " %d total bytes read\n", bytesRead + numBytesRead );
}
// Check that the bytes read have the value of the bytes written
for (i = 0; i < numBytesRead; i++) {
if (bytes[i] != (((bytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE)) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketReader, bytes[%d]=bytes[%d]=%d, expected %d, bytesRead=%d, numBytesRead=%d",
bytesRead + i, i, bytes[i],
(((bytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE),
bytesRead, numBytesRead ));
}
}
for (i = 0; i < numBytesRead; i++) {
ASSERT( bytes[i] == (((bytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE ) );
}
if (error != IMQ_SUCCESS) {
break;
}
bytesRead += numBytesRead;
}
//
// Writer block
//
{
// Initialize bytes
for (int i = 0; i < g_bytesPerIter; i++) {
bytes[i] = (PRUint8)(((bytesWritten + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE);
if ((i == 0) || (i == (g_bytesPerIter - 1))) {
LOG_FINEST(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"writer bytes[%d] = %d",
bytesWritten + i, bytes[i] ));
}
}
//error = sock.write(g_bytesPerIter, bytes, g_timeoutMSWriter, &numBytesWritten);
numBytesWritten = PR_Send(sock.hostSocket, bytes, g_bytesPerIter, 0, 0);
error = IMQ_SUCCESS;
if (numBytesWritten < 0) {
numBytesWritten = 0;
}
if (numBytesWritten > 0) {
fprintf(stderr, "%d total bytes written\n", bytesWritten + numBytesWritten);
}
if (error == (iMQError)PR_WOULD_BLOCK_ERROR) {
}
else if (error != IMQ_SUCCESS) {
break;
}
bytesWritten+= numBytesWritten;
}
}
fprintf(stderr, "Stopping because error = %d\n", error);
}
static void socketReader(void *arg)
{
UNUSED(arg);
static PRUint8 bytes[MAX_BYTES];
PRInt32 numBytesRead = 0;
PRInt32 error = IMQ_SUCCESS;
PRInt32 nextTimeToPrint = 0;
PRInt32 totalBytes = g_bytesPerIter * g_numIters;
for (int bytesRead = 0; bytesRead < totalBytes; ) {
numBytesRead = 0;
bytes[0] = 0;
bytes[1] = 0;
if (g_maxSleepMSReader > 0) {
PRUint32 sleeptime = 1000 * (rand()%g_maxSleepMSReader);
PR_Sleep(PR_MicrosecondsToInterval(sleeptime));
}
// Initialize bytes
int i;
for (i = 0; i < g_bytesPerIter; i++) {
bytes[i] = READ_BYTE;
}
error = sock.read(g_bytesPerIter, g_timeoutMSReader, bytes, &numBytesRead);
if (numBytesRead > 0) {
fprintf(stderr, " %d total bytes read\n", bytesRead + numBytesRead );
}
// Check that the bytes read have the value of the bytes written
for (i = 0; i < numBytesRead; i++) {
if (bytes[i] != (((bytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE)) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketReader, bytes[%d]=bytes[%d]=%d, expected %d, bytesRead=%d, numBytesRead=%d",
bytesRead + i, i, bytes[i],
(((bytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE),
bytesRead, numBytesRead ));
}
}
for (i = 0; i < numBytesRead; i++) {
ASSERT( bytes[i] == (((bytesRead + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE ) );
}
if (numBytesRead == 0) {
PR_Sleep(0);
}
if (error != IMQ_SUCCESS) {
fprintf(stderr, "socket.read returned %d\n", error);
break;
}
bytesRead += numBytesRead;
if (bytesRead > nextTimeToPrint) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketReader, totalBytesRead=%d, thisBytesRead=%d",
bytesRead, numBytesRead ));
//fprintf(stderr, "%d\n", bytesRead);
nextTimeToPrint = bytesRead + g_printInc;
}
// randomly close the socket
if (((double)rand()/(double)RAND_MAX) < g_readerCloseProb) {
fprintf(stderr, "Reader closing the socket early\n" );
sock.close();
break;
}
}
}
/**
*
*/
static void socketWriter(void *arg)
{
UNUSED(arg);
static PRUint8 bytes[MAX_BYTES];
iMQError error = IMQ_SUCCESS;
PRInt32 numBytesWritten = 0;
PRInt32 nextTimeToPrint = 0;
PRInt32 totalBytes = g_bytesPerIter * g_numIters;
for (int bytesWritten = 0; bytesWritten < totalBytes; ) {
if (g_maxSleepMSWriter > 0) {
PRUint32 sleeptime = 1000 * (rand()%g_maxSleepMSWriter);
PR_Sleep(PR_MicrosecondsToInterval(sleeptime));
}
// Initialize bytes
for (int i = 0; i < g_bytesPerIter; i++) {
bytes[i] = (PRUint8)(((bytesWritten + i) / TEST_DIV_VALUE) % TEST_MOD_VALUE);
if ((i == 0) || (i == (g_bytesPerIter - 1))) {
LOG_FINEST(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"writer bytes[%d] = %d",
bytesWritten + i, bytes[i] ));
}
}
error = sock.write(g_bytesPerIter, bytes, g_timeoutMSWriter, &numBytesWritten);
if (numBytesWritten > 0) {
fprintf(stderr, "%d total bytes written\n", bytesWritten + numBytesWritten);
}
if (error == (iMQError)PR_WOULD_BLOCK_ERROR) {
}
else if (error != IMQ_SUCCESS) {
fprintf(stderr, "socket.write returned %d\n", error);
break;
}
bytesWritten+= numBytesWritten;
if (bytesWritten >= nextTimeToPrint) {
// This LOG_WARNING call can include class information since it's only called during testing
LOG_WARNING(( CODELOC, SOCKET_LOG_MASK, NULL_CONN_ID, IMQ_SUCCESS,
"socketWriter, totalBytesWritten=%d, thisBytesWritten=%d",
bytesWritten, numBytesWritten ));
nextTimeToPrint = bytesWritten + g_printInc;
}
// randomly close the socket
if (((double)rand()/(double)RAND_MAX) < g_writerCloseProb) {
fprintf(stderr, "Writer closing the socket early\n");
sock.close();
break;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy