Bug 1187 - Memory leak associated with set_policy_overrides call
Summary: Memory leak associated with set_policy_overrides call
Status: NEW
Alias: None
Product: TAO
Classification: Unclassified
Component: ORB (show other bugs)
Version: 1.2.1
Hardware: x86 Windows 2000
: P3 normal
Assignee: DOC Center Support List (internal)
URL:
Depends on:
Blocks:
 
Reported: 2002-04-16 07:25 CDT by Rob
Modified: 2006-04-21 12:15 CDT (History)
0 users

See Also:


Attachments

Note You need to log in before you can comment on or make changes to this bug.
Description Rob 2002-04-16 07:25:36 CDT
The set_policy_overrides operation returns an object reference.  If a oneway 
call is made on the returned reference, a memory leak occurs.  If a normal call 
is made on the returned reference, no memory leak occurs.  This is all 
according to BoundsChecker.

The following programs will demonstrate the problem:

==== test.idl ====

module JCAFCore {

  interface Listener {

    oneway void update1way();
    void update();

  };

};

==== client_main.cpp ====

#include "testC.h"

class Client
{
  public:
    void init(CORBA::ORB_ptr orb);
    JCAFCore::Listener_ptr setPolicyOverrides(JCAFCore::Listener_ptr listener);

  private:
    CORBA::ORB_ptr theOrb;
};

void
Client::init(CORBA::ORB_ptr orb)
{
  theOrb = orb;
}

JCAFCore::Listener_ptr
Client::setPolicyOverrides(JCAFCore::Listener_ptr listener)
{
  Messaging::SyncScope listenerSyncPolicy(Messaging::SYNC_WITH_TRANSPORT);
  CORBA::PolicyList policyList( 1);
  policyList.length( 1);
  CORBA::Any syncPolicy;
  syncPolicy <<= listenerSyncPolicy;
  policyList[0] = theOrb->create_policy(
      Messaging::SYNC_SCOPE_POLICY_TYPE,
      syncPolicy);
  CORBA::Object_var obj = listener->_set_policy_overrides(
      policyList,
      CORBA::SET_OVERRIDE);
  policyList[0]->destroy();
  return JCAFCore::Listener::_narrow( obj.in());
}

int
main (int argc, char *argv[])
{
  ACE_DECLARE_NEW_CORBA_ENV;
  ACE_TRY
    {
      CORBA::ORB_var orb =
        CORBA::ORB_init (argc, argv, 0, ACE_TRY_ENV);
      ACE_TRY_CHECK;

      Client client;
      client.init(orb);

      std::ifstream file;
      file.open( "c:\\ior.txt");
      std::string iorString;
      file >> iorString;

      CORBA::Object_var obj = orb->string_to_object( iorString.c_str());

      JCAFCore::Listener_var listener = JCAFCore::Listener::_narrow(obj.in());

      // There is no leak that shows up 77 times.
      for (int i=0; i<77; i++)
        {
          JCAFCore::Listener_var ref = client.setPolicyOverrides(listener.in());
          ref->update();
        }

      // There is a memory leak on set_policy_overrides that shows up 99 times.
      for (int i=0; i<99; i++)
        {
          JCAFCore::Listener_var ref = client.setPolicyOverrides(listener.in());
          ref->update1way();
        }

      orb->destroy();

    }
  ACE_CATCHANY
    {
      ACE_PRINT_EXCEPTION (ACE_ANY_EXCEPTION,"main");
      return -1;
    }
  ACE_ENDTRY;
  ACE_CHECK_RETURN (-1);

  return 0;
}

==== server_main.cpp ====

#include "testS.h"

class ListenerImpl : public POA_JCAFCore::Listener
{
  public:
    virtual void update ()
      ACE_THROW_SPEC (( CORBA::SystemException ));
    virtual void update1way ()
      ACE_THROW_SPEC (( CORBA::SystemException ));
};


void ListenerImpl::update ( )
    ACE_THROW_SPEC (( CORBA::SystemException ))
{
  ACE_DEBUG((LM_INFO, "update called\n"));
}

void
ListenerImpl::update1way ()
    ACE_THROW_SPEC (( CORBA::SystemException ))
{
  ACE_DEBUG(( LM_INFO, "update1way called\n"));
}


int
main (int argc, char *argv[])
{
  ACE_DECLARE_NEW_CORBA_ENV;
  ACE_TRY
    {
      CORBA::ORB_var orb =
        CORBA::ORB_init (argc, argv, 0, ACE_TRY_ENV);
      ACE_TRY_CHECK;

      CORBA::Object_var obj =
          orb->resolve_initial_references ("RootPOA", ACE_TRY_ENV);
      ACE_TRY_CHECK;

      PortableServer::POA_var root_poa =
          PortableServer::POA::_narrow (obj.in (), ACE_TRY_ENV);
      ACE_TRY_CHECK;

      PortableServer::POAManager_var mgr =
          root_poa->the_POAManager (ACE_TRY_ENV);
      ACE_TRY_CHECK;

      mgr->activate (ACE_TRY_ENV);
      ACE_TRY_CHECK;

      ListenerImpl server;

      obj = server._this();

      std::ofstream file;
      file.open( "c:\\ior.txt");

      CORBA::String_var iorString = orb->object_to_string( obj);

      file << iorString;
      file.close();

      orb->run();

      orb->destroy();

    }
  ACE_CATCHANY
    {
      ACE_PRINT_EXCEPTION (ACE_ANY_EXCEPTION,"main");
      return -1;
    }
  ACE_ENDTRY;
  ACE_CHECK_RETURN (-1);

  return 0;
}
Comment 1 Ossama Othman 2002-04-16 13:00:03 CDT
Reassign to myself.
Comment 2 Ossama Othman 2002-04-16 13:00:16 CDT
Mine.
Comment 3 Ossama Othman 2002-04-16 16:24:57 CDT
Rob, can you tell me where BoundsChecker reports the leak?  I want to make 
sure I got the leak you're reporting.  My BoundsChecker installation is 
currently broken due to an upgrade to XP, and I haven't received my new copy 
of BoundsChecker yet.
Comment 4 Ossama Othman 2002-04-18 12:14:52 CDT
The following is Rob's stack trace:

TAO_Sync_Scope_Policy::copy()   .\Messaging_Policy_i.cpp        285
TAO_Policy_Set::set_policy()    .\Policy_Set.cpp        191
TAO_Policy_Set::set_policy_overrides()  .\Policy_Set.cpp        176
TAO_Stub::set_policy_overrides()        .\Stub.cpp      488
CORBA_Object::_set_policy_overrides()   .\Object.cpp    432
Client::setPolicyOverrides()    C:\client_main.cpp      30

Thanks Rob!
Comment 5 Ossama Othman 2002-04-18 13:01:59 CDT
Rob, I'm going to need a different stack trace.  In particular, BoundsChecker 
should show a stack trace that shows exactly where the leak occurred during 
the oneway invocation.  The stack trace you gave me is for the code you 
execute prior to your oneway invocation.  (Respond via the bugzilla interface 
on the web, not via e-mail.  Otherwise I won't get the response.)
Comment 6 Rob 2002-04-18 13:15:33 CDT
BoundsChecker is only giving me the location of the memory leak.  It doesn't 
refer to the oneway call at all.  The bottom line in the trace I sent is 
referring to this operation:

CORBA::Policy_ptr
TAO_Sync_Scope_Policy::copy (CORBA_Environment &ACE_TRY_ENV)
  ACE_THROW_SPEC ((CORBA::SystemException))
{
  TAO_Sync_Scope_Policy *servant = 0;
  ACE_NEW_THROW_EX (servant,
                    TAO_Sync_Scope_Policy (*this),
                    CORBA::NO_MEMORY ());
  ACE_CHECK_RETURN (CORBA::Policy::_nil ());

  return servant;
}

The ACE_NEW_THROW_EX line is being flagged so I assume it is saying that 
the "servant" created here is being leaked.
Comment 7 Ossama Othman 2002-04-18 13:41:35 CDT
It's possible that this might be a false-positive on BoundsChecker's part.  I 
can't see how the copy() method leaks since nothing occurs after the 
allocation.  The return value of the copy() method is stored in a Policy_var, 
after which proper memory management appears to occur.

The code in question is independent of whether or not the invocation is a 
twoway or a oneway, which is what leads me to believe that if there is a leak 
for oneways it must be in the oneway critical path.  I'll try to run your code 
with Purify on Solaris to see if I get leaks for oneway invocations.

Out of curiosity, what happens if you change your client loops to:

     JCAFCore::Listener_var ref = client.setPolicyOverrides(listener.in());

     // There is no leak that shows up 77 times.
      for (int i=0; i<77; i++)
        {
          ref->update();
        }

      // There is a memory leak on set_policy_overrides that shows up 99 times.
      for (int i=0; i<99; i++)
        {
          ref->update1way();
        }

Presumably you should still get leaks.
Comment 8 Rob 2002-04-18 14:01:31 CDT
I forgot to mention that.  The reason I put the set_policy_overrides call in 
the loops is so it was obvious when there was a leak since it happened so many 
times.  If you move the call outside the loop, there is only a single memory 
leak.