summaryrefslogtreecommitdiffstats
path: root/gdbstub.c
Commit message (Expand)AuthorAgeFilesLines
* gdbstub: ensure we clean-up when terminatedAlex Bennée2021-01-181-0/+2
* gdbstub: drop gdbserver_cleanup in favour of gdb_exitAlex Bennée2021-01-181-7/+0Star
* gdbstub: drop CPUEnv from gdb_exit()Alex Bennée2021-01-181-1/+1
* gdbstub: add support to Xfer:auxv:read: packetLirong Yuan2021-01-181-0/+54
* gdbstub: Correct misparsing of vCont C/S requestsPeter Maydell2020-12-151-1/+1
* replay: create temporary snapshot at debugger connectionPavel Dovgalyuk2020-10-061-0/+1
* gdbstub: add reverse continue support in replay modePavel Dovgalyuk2020-10-061-1/+9
* gdbstub: add reverse step support in replay modePavel Dovgalyuk2020-10-061-2/+53
* trace: switch position of headers to what Meson requiresPaolo Bonzini2020-08-211-1/+1
* qom: Change object_get_canonical_path_component() not to mallocMarkus Armbruster2020-07-211-1/+1
* gdbstub/linux-user: support debugging over a unix socketAlex Bennée2020-05-061-16/+87
* gdbstub: eliminate gdbserver_fd globalAlex Bennée2020-05-061-13/+11Star
* gdbstub: fix compiler complainingDenis Plotnikov2020-04-071-2/+2
* gdbstub: Fix single-step issue by confirming 'vContSupported+' feature to gdbChangbin Du2020-03-171-1/+1
* gdbstub: do not split gdb_monitor_write payloadDamien Hedde2020-03-171-20/+3Star
* gdbstub: change GDBState.last_packet to GByteArrayDamien Hedde2020-03-171-18/+21
* gdbstub: extend GByteArray to read register helpersAlex Bennée2020-03-171-10/+10
* gdbstub: move mem_buf to GDBState and use GByteArrayAlex Bennée2020-03-171-21/+35
* gdbstub: move str_buf to GDBState and use GStringAlex Bennée2020-03-171-105/+90Star
* gdbstub: stop passing GDBState * around and use globalAlex Bennée2020-03-171-283/+278Star
* gdbstub: make GDBState static and have common init functionAlex Bennée2020-03-171-87/+81Star
* hmp: Fail gracefully if chardev is already in useKevin Wolf2020-03-061-1/+1
* chardev: Use QEMUChrEvent enum in IOEventHandler typedefPhilippe Mathieu-Daudé2020-01-081-1/+1
* gdbstub: Fix handler for 'F' packetSandra Loosemore2019-08-281-2/+6
* gdbstub: Fix handling of '!' packet with new infraRamiro Polla2019-08-281-1/+3
* sysemu: Split sysemu/runstate.h off sysemu/sysemu.hMarkus Armbruster2019-08-161-0/+1
* gdbstub: revert to previous set_reg behaviourAlex Bennée2019-07-101-6/+12
* gdbstub: add some notes to the header commentAlex Bennée2019-07-101-0/+6
* general: Replace global smp variables with smp machine propertiesLike Xu2019-07-051-0/+4
* monitor: Replace monitor_init() with monitor_init_{hmp, qmp}()Kevin Wolf2019-06-181-1/+1
* gdbstub: Implement qemu physical memory modeJon Doron2019-06-121-2/+60
* gdbstub: Clear unused variables in gdb_handle_packetJon Doron2019-06-121-9/+2Star
* gdbstub: Implement target halted (? pkt) with new infraJon Doron2019-06-121-10/+25
* gdbstub: Implement generic set/query (Q/q pkt) with new infraJon Doron2019-06-121-186/+373
* gdbstub: Implement v commands with new infraJon Doron2019-06-121-60/+110
* gdbstub: Implement step (s pkt) with new infraJon Doron2019-06-121-6/+19
* gdbstub: Implement file io (F pkt) with new infraJon Doron2019-06-121-22/+26
* gdbstub: Implement read all registers (g pkt) with new infraJon Doron2019-06-121-8/+23
* gdbstub: Implement write all registers (G pkt) with new infraJon Doron2019-06-121-10/+31
* gdbstub: Implement read memory (m pkt) with new infraJon Doron2019-06-121-16/+32
* gdbstub: Implement write memory (M pkt) with new infraJon Doron2019-06-121-18/+33
* gdbstub: Implement get register (p pkt) with new infraJon Doron2019-06-121-12/+38
* gdbstub: Implement set register (P pkt) with new infraJon Doron2019-06-121-9/+30
* gdbstub: Implement breakpoint commands (Z/z pkt) with new infraJon Doron2019-06-121-19/+67
* gdbstub: Implement set_thread (H pkt) with new infraJon Doron2019-06-121-30/+53
* gdbstub: Implement continue with signal (C pkt) with new infraJon Doron2019-06-121-5/+29
* gdbstub: Implement continue (c pkt) with new infraJon Doron2019-06-121-6/+19
* gdbstub: Implement thread_alive (T pkt) with new infraJon Doron2019-06-121-11/+32
* gdbstub: Implement deatch (D pkt) with new infraJon Doron2019-06-121-40/+61
* gdbstub: Add infrastructure to parse cmd packetsJon Doron2019-06-121-0/+195
class='oid'>cd9ba1ebcf ^
7c0628b20e ^
a76bab4952 ^
9eb0bfca96 ^
f42b22077b ^
164a101f28 ^
f42b22077b ^
438e1f47e7 ^
7c0628b20e ^

a76bab4952 ^
8febfa2684 ^

bcdc18578d ^
87f68d3182 ^
8febfa2684 ^
a915f4bc97 ^
7c0628b20e ^



438e1f47e7 ^


cd9ba1ebcf ^


87f68d3182 ^
cd9ba1ebcf ^




cd9ba1ebcf ^


f42b22077b ^


164a101f28 ^

8b2d42d273 ^
164a101f28 ^

cd9ba1ebcf ^












7c0628b20e ^
bcdc18578d ^
bafbd6a1c6 ^
8febfa2684 ^
a915f4bc97 ^
a76bab4952 ^
9eb0bfca96 ^
f42b22077b ^
a915f4bc97 ^
f42b22077b ^

9eb0bfca96 ^

a76bab4952 ^
a915f4bc97 ^
a76bab4952 ^
9eb0bfca96 ^
b022b4a44a ^
438e1f47e7 ^




f42b22077b ^






9eb0bfca96 ^
9eb0bfca96 ^
87f68d3182 ^
a915f4bc97 ^
9eb0bfca96 ^


a915f4bc97 ^
2db2bfc0cc ^
9eb0bfca96 ^
f42b22077b ^


164a101f28 ^

8b2d42d273 ^
164a101f28 ^

a76bab4952 ^

9eb0bfca96 ^


a915f4bc97 ^
2db2bfc0cc ^
a915f4bc97 ^
9eb0bfca96 ^


a76bab4952 ^
b022b4a44a ^


9eb0bfca96 ^
bcdc18578d ^
438e1f47e7 ^








164a101f28 ^
a76bab4952 ^
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223


                          

                               


                                           
                                           



                                                                        

                                                                     


                        
                        

                         
 


                                    
                
                
                                 

  

                                             
                                                            


                     
                                                   


                                             

     
                                         
                     
                   

                                                           
                                                                    
                                        
                                  

                                      



                                                                           
                                         
                             




                                                            
                                                 


                                                                   
                                                              

                                                        

                                                    
                                    
     

                    

 




                                                   
                                                   






                        
                                             
 
                     
                                            
                  
              
                

                     
 

                                                                               
                                                                              
                                                                          
       
                           



                         


                                                     


                                                            
                                                                   




                                           


                                


                                                   

                                                         
                                            

                                












                                                     
                                
                    
     
 
                            
 
                      
              
                                                   

                                                                 

         
 
                            
 
                               
                       




                                                                             






                                                            
 
                                                 
                                                              
                                               


                            
                                    
 
                                 


                                                                                    

                                                             
                                                

                                    

             


                                          
                                    
 
                                                         


                                        
         


                                                          
     
 








                                                                              
                    
 
/*
 * QEMU aio implementation
 *
 * Copyright IBM Corp., 2008
 * Copyright Red Hat Inc., 2012
 *
 * Authors:
 *  Anthony Liguori   <aliguori@us.ibm.com>
 *  Paolo Bonzini     <pbonzini@redhat.com>
 *
 * This work is licensed under the terms of the GNU GPL, version 2.  See
 * the COPYING file in the top-level directory.
 *
 * Contributions after 2012-01-13 are licensed under the terms of the
 * GNU GPL, version 2 or (at your option) any later version.
 */

#include "qemu-common.h"
#include "block/block.h"
#include "qemu/queue.h"
#include "qemu/sockets.h"

struct AioHandler {
    EventNotifier *e;
    EventNotifierHandler *io_notify;
    GPollFD pfd;
    int deleted;
    QLIST_ENTRY(AioHandler) node;
};

void aio_set_event_notifier(AioContext *ctx,
                            EventNotifier *e,
                            EventNotifierHandler *io_notify)
{
    AioHandler *node;

    QLIST_FOREACH(node, &ctx->aio_handlers, node) {
        if (node->e == e && !node->deleted) {
            break;
        }
    }

    /* Are we deleting the fd handler? */
    if (!io_notify) {
        if (node) {
            g_source_remove_poll(&ctx->source, &node->pfd);

            /* If the lock is held, just mark the node as deleted */
            if (ctx->walking_handlers) {
                node->deleted = 1;
                node->pfd.revents = 0;
            } else {
                /* Otherwise, delete it for real.  We can't just mark it as
                 * deleted because deleted nodes are only cleaned up after
                 * releasing the walking_handlers lock.
                 */
                QLIST_REMOVE(node, node);
                g_free(node);
            }
        }
    } else {
        if (node == NULL) {
            /* Alloc and insert if it's not already there */
            node = g_malloc0(sizeof(AioHandler));
            node->e = e;
            node->pfd.fd = (uintptr_t)event_notifier_get_handle(e);
            node->pfd.events = G_IO_IN;
            QLIST_INSERT_HEAD(&ctx->aio_handlers, node, node);

            g_source_add_poll(&ctx->source, &node->pfd);
        }
        /* Update handler with latest information */
        node->io_notify = io_notify;
    }

    aio_notify(ctx);
}

bool aio_pending(AioContext *ctx)
{
    AioHandler *node;

    QLIST_FOREACH(node, &ctx->aio_handlers, node) {
        if (node->pfd.revents && node->io_notify) {
            return true;
        }
    }

    return false;
}

bool aio_poll(AioContext *ctx, bool blocking)
{
    AioHandler *node;
    HANDLE events[MAXIMUM_WAIT_OBJECTS + 1];
    bool progress;
    int count;
    int timeout;

    progress = false;

    /*
     * If there are callbacks left that have been queued, we need to call then.
     * Do not call select in this case, because it is possible that the caller
     * does not need a complete flush (as is the case for aio_poll loops).
     */
    if (aio_bh_poll(ctx)) {
        blocking = false;
        progress = true;
    }

    /* Run timers */
    progress |= timerlistgroup_run_timers(&ctx->tlg);

    /*
     * Then dispatch any pending callbacks from the GSource.
     *
     * We have to walk very carefully in case aio_set_fd_handler is
     * called while we're walking.
     */
    node = QLIST_FIRST(&ctx->aio_handlers);
    while (node) {
        AioHandler *tmp;

        ctx->walking_handlers++;

        if (node->pfd.revents && node->io_notify) {
            node->pfd.revents = 0;
            node->io_notify(node->e);

            /* aio_notify() does not count as progress */
            if (node->e != &ctx->notifier) {
                progress = true;
            }
        }

        tmp = node;
        node = QLIST_NEXT(node, node);

        ctx->walking_handlers--;

        if (!ctx->walking_handlers && tmp->deleted) {
            QLIST_REMOVE(tmp, node);
            g_free(tmp);
        }
    }

    if (progress && !blocking) {
        return true;
    }

    ctx->walking_handlers++;

    /* fill fd sets */
    count = 0;
    QLIST_FOREACH(node, &ctx->aio_handlers, node) {
        if (!node->deleted && node->io_notify) {
            events[count++] = event_notifier_get_handle(node->e);
        }
    }

    ctx->walking_handlers--;

    /* wait until next event */
    while (count > 0) {
        int ret;

        timeout = blocking ?
            qemu_timeout_ns_to_ms(timerlistgroup_deadline_ns(&ctx->tlg)) : 0;
        ret = WaitForMultipleObjects(count, events, FALSE, timeout);

        /* if we have any signaled events, dispatch event */
        if ((DWORD) (ret - WAIT_OBJECT_0) >= count) {
            break;
        }

        blocking = false;

        /* we have to walk very carefully in case
         * aio_set_fd_handler is called while we're walking */
        node = QLIST_FIRST(&ctx->aio_handlers);
        while (node) {
            AioHandler *tmp;

            ctx->walking_handlers++;

            if (!node->deleted &&
                event_notifier_get_handle(node->e) == events[ret - WAIT_OBJECT_0] &&
                node->io_notify) {
                node->io_notify(node->e);

                /* aio_notify() does not count as progress */
                if (node->e != &ctx->notifier) {
                    progress = true;
                }
            }

            tmp = node;
            node = QLIST_NEXT(node, node);

            ctx->walking_handlers--;

            if (!ctx->walking_handlers && tmp->deleted) {
                QLIST_REMOVE(tmp, node);
                g_free(tmp);
            }
        }

        /* Try again, but only call each handler once.  */
        events[ret - WAIT_OBJECT_0] = events[--count];
    }

    if (blocking) {
        /* Run the timers a second time. We do this because otherwise aio_wait
         * will not note progress - and will stop a drain early - if we have
         * a timer that was not ready to run entering g_poll but is ready
         * after g_poll. This will only do anything if a timer has expired.
         */
        progress |= timerlistgroup_run_timers(&ctx->tlg);
    }

    return progress;
}