English Amiga Board


Go Back   English Amiga Board > Coders > Coders. System

 
 
Thread Tools
Old 27 October 2018, 13:31   #1
thyslo
Registered User
 
Join Date: Apr 2018
Location: Germany
Posts: 189
Help needed for an Intuition event loop

Hi all,
I’m relatively new to Amiga programming, nevertheless I choose a bigger project coded in c++ to start with it.

And now I’m messing around in the Intuition event loop without getting any further. Maybe one of you more advanced Amiga programmers see the fault and can give a hint?

My program has up to three windows opened. But at the start only one window is opened. When entering the loop only pWin3 is opened, pWin1 and pWin2 are NULL. Here’s the loop as it is by now:

Code:
 1 void Application::intuiEventLoop()
 2 {
 3   struct Window* pWin1 = m_pLeftWin->IntuiWindow();
 4   struct Window* pWin2 = m_pRightWin->IntuiWindow();
 5   struct Window* pWin3 = m_pOpenFilesWin->IntuiWindow();
 6   struct Menu* pMenu = m_pMenu->IntuiMenu();
 7 
 8   while(m_bExitRequested == false)
 9   {
10     // Waiting for messages from Intuition
11     Wait(1L << pWin1->UserPort->mp_SigBit |
12          1L << pWin2->UserPort->mp_SigBit |
13          1L << pWin3->UserPort->mp_SigBit);
14 
15     struct IntuiMessage* pMsg;
16 
17     while ((pMsg = (struct IntuiMessage *)GetMsg(pWin1->UserPort)) ||
18            (pMsg = (struct IntuiMessage *)GetMsg(pWin2->UserPort)) ||
19            (pMsg = (struct IntuiMessage *)GetMsg(pWin3->UserPort)) )
20     {
21       // Get all data we need from message
22       ULONG msgClass = pMsg->Class;
23       UWORD msgCode = pMsg->Code;
24       APTR msgIAddress = pMsg->IAddress;
25       struct Window* msgWindow = pMsg->IDCMPWindow;
26 
27       // When we're through with a message, reply
28       ReplyMsg((struct Message *)pMsg);
29 
30       if(msgClass == IDCMP_MENUPICK)
31       {
32         //
33         // Menupick messages are handled here
34         //
35         UWORD menuNumber = msgCode;
36         struct MenuItem* pSelectedItem = ItemAddress(pMenu, menuNumber);
37 
38         if(pSelectedItem != NULL)
39         {
40           // Getting the user data from selected menu item
41           APTR pUserData = GTMENUITEM_USERDATA(pSelectedItem);
42           if(pUserData != NULL)
43           {
44             // Our menu user data always contain a pointer to a Command
45             Command* pSelecedCommand = static_cast<Command*>(pUserData);
46 
47             // Execute this command
48             pSelecedCommand->Execute();
49           }
50         }
51       }
52       else
53       {
54         //
55         // All other messages are handled in the appropriate window
56         //
57         if(msgWindow == m_pLeftWin->IntuiWindow())
58         {
59           m_pLeftWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
60         }
61         else if(msgWindow == m_pRightWin->IntuiWindow())
62         {
63           m_pRightWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
64         }
65         else if(msgWindow == m_pOpenFilesWin->IntuiWindow())
66         {
67           m_pOpenFilesWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
68         }
69       }
70     }
71   }
72 }
The problem is, the program crashes if I close pWin3 by clicking on its close icon.

By clicking on the close icon the HandleIdcmp() method of the higher level c++ object encapsulation of pWin3 gets called. There inside when the close icon was clicked, CloseWindow() will be called. So after this the pointer pWin3 points to an address where no opened window exists anymore.

Can this be the reason for the crash?

Or is it the Wait() call where an ->UserPort->mp_SigBit of the NULL pointers is done, as pWin1 and pWin2 are NULL at the beginning.

But even if I comment out all pWin1 and pWin2 occurrences inside the loop the program crashes.

Is there a better way to handle multiple windows in a higher level event loop without making it too complex?

The complete src, coded in C++ using StormC4 can be found here: https://github.com/rosneru/AmiDiff
thyslo is online now  
Old 27 October 2018, 15:02   #2
alkis
Registered User
 
Join Date: Dec 2010
Location: Athens/Greece
Age: 53
Posts: 719
When you press the close button on the window, you actually close the window and then the (pMsg = (struct IntuiMessage *)GetMsg(pWin3->UserPort)) is accessing something that is not there.
alkis is offline  
Old 27 October 2018, 16:10   #3
thyslo
Registered User
 
Join Date: Apr 2018
Location: Germany
Posts: 189
Hi alkis, thanks your answer!

This can indeed be a problem. So I've changed the code so that GetMsg() now hopefully is called with valid arguments. Sadly it still crashes. This is the changed event loop:

Code:
void Application::intuiEventLoop()
{
  struct Window* pWin1 = m_pLeftWin->IntuiWindow();
  struct Window* pWin2 = m_pRightWin->IntuiWindow();
  struct Window* pWin3 = m_pOpenFilesWin->IntuiWindow();

  struct Menu* pMenu = m_pMenu->IntuiMenu();

  while((pWin1 != NULL || pWin2 != NULL || pWin3 != NULL) &&
        m_bExitRequested == false)
  {
    // Waiting for messages from Intuition
    ULONG mask = 0;
    if(m_pLeftWin != NULL && m_pLeftWin->IntuiWindow() != NULL)
    {
      mask |= 1L << m_pLeftWin->IntuiWindow()->UserPort->mp_SigBit;
    }

    if(m_pRightWin != NULL && m_pRightWin->IntuiWindow() != NULL)
    {
      mask |= 1L << m_pRightWin->IntuiWindow()->UserPort->mp_SigBit;
    }

    if(m_pOpenFilesWin != NULL && m_pOpenFilesWin->IntuiWindow() != NULL)
    {
      mask |= 1L << m_pOpenFilesWin->IntuiWindow()->UserPort->mp_SigBit;
    }

    Wait(mask);

    struct IntuiMessage* pMsg;

    while ((pWin1 != NULL) && (pMsg = (struct IntuiMessage *)GetMsg(pWin1->UserPort)) ||
           (pWin2 != NULL) && (pMsg = (struct IntuiMessage *)GetMsg(pWin2->UserPort)) ||
           (pWin3 != NULL) && (pMsg = (struct IntuiMessage *)GetMsg(pWin3->UserPort)))
    {
      // Get all data we need from message
      ULONG msgClass = pMsg->Class;
      UWORD msgCode = pMsg->Code;
      APTR msgIAddress = pMsg->IAddress;
      struct Window* msgWindow = pMsg->IDCMPWindow;

      // When we're through with a message, reply
      ReplyMsg((struct Message *)pMsg);

      if(msgClass == IDCMP_MENUPICK)
      {
        //
        // Menupick messages are handled here
        //
        UWORD menuNumber = msgCode;
        struct MenuItem* pSelectedItem = ItemAddress(pMenu, menuNumber);

        if(pSelectedItem != NULL)
        {
          // Getting the user data from selected menu item
          APTR pUserData = GTMENUITEM_USERDATA(pSelectedItem);
          if(pUserData != NULL)
          {
            // Our menu user data always contain a pointer to a Command
            Command* pSelecedCommand = static_cast<Command*>(pUserData);

            // Execute this command
            pSelecedCommand->Execute();
          }
        }
      }
      else
      {
        //
        // All other messages are handled in the appropriate window
        //
        if(msgWindow == m_pLeftWin->IntuiWindow())
        {
          m_pLeftWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
        }
        else if(msgWindow == m_pRightWin->IntuiWindow())
        {
          m_pRightWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
        }
        else if(msgWindow == m_pOpenFilesWin->IntuiWindow())
        {
          m_pOpenFilesWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
        }
      }

      pWin1 = m_pLeftWin->IntuiWindow();
      pWin2 = m_pRightWin->IntuiWindow();
      pWin3 = m_pOpenFilesWin->IntuiWindow();

      if(pWin1 == NULL && pWin2 == NULL && pWin3 == NULL)
      {
        // Exit because all windows are closed
        m_bExitRequested = true;
      }
    }
  }
}
With StormC debugger i can step trough the program. The crash occurs after the final return 0; from main().

Could it be that in my loop I forgot to reply a message? Could this lead to a crash?
thyslo is online now  
Old 27 October 2018, 19:16   #4
alkis
Registered User
 
Join Date: Dec 2010
Location: Athens/Greece
Age: 53
Posts: 719
I think you are still executing the inner loop.

Try adding a break to get out of the inner loop.
Code:
if(pWin1 == NULL && pWin2 == NULL && pWin3 == NULL)
      {
        // Exit because all windows are closed
        m_bExitRequested = true;
        break;      /* <---- add this */
      }
alkis is offline  
Old 28 October 2018, 04:28   #5
Hewitson
Registered User
 
Hewitson's Avatar
 
Join Date: Feb 2007
Location: Melbourne, Australia
Age: 41
Posts: 3,771
If you don't mind me saying, for such a simple program, it's quite unreadable. May I suggest using better indenting/formatting techniques? It will make your code much easier to debug.
Hewitson is offline  
Old 28 October 2018, 17:27   #6
thyslo
Registered User
 
Join Date: Apr 2018
Location: Germany
Posts: 189
I've found a solution that seems to work. I've introduced two helper methods for generating the signal mask and for getting the next message from any of the opening windows. This also makes the code a little better readable.

Current event loop and the two helper methods:

Code:
void Application::intuiEventLoop()
{
  ULONG mask;
  struct IntuiMessage* pMsg;
  do
  {
    mask = signalMask();
    Wait(mask);
    while (pMsg = nextIntuiMessage())
    {
      // Get all data we need from message
      ULONG msgClass = pMsg->Class;
      UWORD msgCode = pMsg->Code;
      APTR msgIAddress = pMsg->IAddress;
      struct Window* msgWindow = pMsg->IDCMPWindow;

      // When we're through with a message, reply
      ReplyMsg((struct Message *)pMsg);

      if(msgClass == IDCMP_MENUPICK)
      {
        //
        // Menupick messages are handled here
        //
        UWORD menuNumber = msgCode;
        struct MenuItem* pSelectedItem = ItemAddress(m_pMenu->IntuiMenu(), menuNumber);

        if(pSelectedItem != NULL)
        {
          // Getting the user data from selected menu item
          APTR pUserData = GTMENUITEM_USERDATA(pSelectedItem);
          if(pUserData != NULL)
          {
            // Our menu user data always contain a pointer to a Command
            Command* pSelecedCommand = static_cast<Command*>(pUserData);

            // Execute this command
            pSelecedCommand->Execute();
          }
        }
      }
      else
      {
        //
        // All other messages are handled in the appropriate window
        //
        if(msgWindow == m_pLeftWin->IntuiWindow())
        {
          m_pLeftWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
        }
        else if(msgWindow == m_pRightWin->IntuiWindow())
        {
          m_pRightWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
        }
        else if(msgWindow == m_pOpenFilesWin->IntuiWindow())
        {
          m_pOpenFilesWin->HandleIdcmp(msgClass, msgCode, msgIAddress);
        }
      }

      if(!m_pLeftWin->IsOpen() && !m_pRightWin->IsOpen() &&
         !m_pOpenFilesWin->IsOpen())
      {
        // All windows are close: exit
        m_bExitRequested = true;
        break;
      }
    }
  }
  while(!m_bExitRequested);
}


ULONG Application::signalMask()
{
  ULONG signal = 0;

  if(m_pLeftWin->IsOpen())
  {
    signal |= 1L << m_pLeftWin->IntuiWindow()->UserPort->mp_SigBit;
  }

  if(m_pRightWin->IsOpen())
  {
    signal |= 1L << m_pRightWin->IntuiWindow()->UserPort->mp_SigBit;
  }

  if(m_pOpenFilesWin->IsOpen())
  {
    signal |= 1L << m_pOpenFilesWin->IntuiWindow()->UserPort->mp_SigBit;
  }

  return signal;
}


struct IntuiMessage* Application::nextIntuiMessage()
{
  struct IntuiMessage* pIntuiMessage = NULL;

  // Look for message in userport of left window
  if(m_pLeftWin->IsOpen())
  {
    pIntuiMessage = (struct IntuiMessage *)
      GetMsg(m_pLeftWin->IntuiWindow()->UserPort);

    if(pIntuiMessage != NULL)
    {
      return pIntuiMessage;
    }
  }

  // Look for message in userport of right window
  if(m_pRightWin->IsOpen())
  {
    pIntuiMessage = (struct IntuiMessage *)
      GetMsg(m_pRightWin->IntuiWindow()->UserPort);

    if(pIntuiMessage != NULL)
    {
      return pIntuiMessage;
    }
  }

    // Look for message in userport of open files window
  if(m_pOpenFilesWin->IsOpen())
  {
    pIntuiMessage = (struct IntuiMessage *)
      GetMsg(m_pOpenFilesWin->IntuiWindow()->UserPort);

    if(pIntuiMessage != NULL)
    {
      return pIntuiMessage;
    }
  }

  return NULL;
}
Quote:
If you don't mind me saying, for such a simple program, it's quite unreadable. May I suggest using better indenting/formatting techniques? It will make your code much easier to debug.
I know what you mean. But I've a rule to myself to preferably not extend a line width of 72 chars. This leads to instructions/lines that go over two or more lines.
thyslo is online now  
Old 28 October 2018, 22:12   #7
thomas
Registered User
 
thomas's Avatar
 
Join Date: Jan 2002
Location: Germany
Posts: 6,985
You should consider to share the same message port among all three windows. Would make your code much smaller and easier.
thomas is offline  
Old 29 October 2018, 08:37   #8
thyslo
Registered User
 
Join Date: Apr 2018
Location: Germany
Posts: 189
Yeah I will look into this and maybe changing it. Yesterday I've found the example "CloseWindowSafely" in the autodocs which let me think about it.
thyslo is online now  
Old 29 October 2018, 19:08   #9
thyslo
Registered User
 
Join Date: Apr 2018
Location: Germany
Posts: 189
Implemented it now with a shared message port, and it works like a charm:-) And is by far more readable.

Thank you all for your help and hints!
thyslo is online now  
 


Currently Active Users Viewing This Thread: 1 (0 members and 1 guests)
 
Thread Tools

Similar Threads
Thread Thread Starter Forum Replies Last Post
C Intuition Help plasmab Coders. General 7 15 October 2018 09:05
WinUAE stuck in a loop Leandro Jardim support.WinUAE 2 07 April 2018 21:38
It Came from the Desert crashing in a Loop gb_pdx support.FS-UAE 5 04 January 2017 20:43
Copper loop confusions Fell Coders. Asm / Hardware 15 21 September 2015 21:02
Sampled loop in cracktro absence request.Music 2 30 June 2012 11:33

Posting Rules
You may not post new threads
You may not post replies
You may not post attachments
You may not edit your posts

BB code is On
Smilies are On
[IMG] code is On
HTML code is Off

Forum Jump


All times are GMT +2. The time now is 10:04.

Top

Powered by vBulletin® Version 3.8.11
Copyright ©2000 - 2024, vBulletin Solutions Inc.
Page generated in 0.22521 seconds with 13 queries