net: fix unbounded NetQueue
In the current implementation of qemu, running without a network backend will cause the queue to grow unbounded when the guest is transmitting traffic. This patch fixes the problem by implementing bounded size NetQueue, used with an arbitrary limit of 10000 packets, and dropping packets when the queue is full _and_ the sender does not pass a callback. The second condition makes sure that we never drop packets that contains a callback (which would be tricky, because the producer expects the callback to be run when all previous packets have been consumed; so we cannot run it when the packet is dropped). If documentation is correct, producers that submit a callback should stop sending when their packet is queued, so there is no real risk that the queue exceeds the max size by large values. Signed-off-by: Luigi Rizzo <rizzo@iet.unipi.it> Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
This commit is contained in:
		
				
					committed by
					
						 Stefan Hajnoczi
						Stefan Hajnoczi
					
				
			
			
				
	
			
			
			
						parent
						
							199ee608f0
						
					
				
				
					commit
					7d91ddd25e
				
			
							
								
								
									
										15
									
								
								net/queue.c
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								net/queue.c
									
									
									
									
									
								
							| @@ -50,6 +50,8 @@ struct NetPacket { | |||||||
|  |  | ||||||
| struct NetQueue { | struct NetQueue { | ||||||
|     void *opaque; |     void *opaque; | ||||||
|  |     uint32_t nq_maxlen; | ||||||
|  |     uint32_t nq_count; | ||||||
|  |  | ||||||
|     QTAILQ_HEAD(packets, NetPacket) packets; |     QTAILQ_HEAD(packets, NetPacket) packets; | ||||||
|  |  | ||||||
| @@ -63,6 +65,8 @@ NetQueue *qemu_new_net_queue(void *opaque) | |||||||
|     queue = g_malloc0(sizeof(NetQueue)); |     queue = g_malloc0(sizeof(NetQueue)); | ||||||
|  |  | ||||||
|     queue->opaque = opaque; |     queue->opaque = opaque; | ||||||
|  |     queue->nq_maxlen = 10000; | ||||||
|  |     queue->nq_count = 0; | ||||||
|  |  | ||||||
|     QTAILQ_INIT(&queue->packets); |     QTAILQ_INIT(&queue->packets); | ||||||
|  |  | ||||||
| @@ -92,6 +96,9 @@ static void qemu_net_queue_append(NetQueue *queue, | |||||||
| { | { | ||||||
|     NetPacket *packet; |     NetPacket *packet; | ||||||
|  |  | ||||||
|  |     if (queue->nq_count >= queue->nq_maxlen && !sent_cb) { | ||||||
|  |         return; /* drop if queue full and no callback */ | ||||||
|  |     } | ||||||
|     packet = g_malloc(sizeof(NetPacket) + size); |     packet = g_malloc(sizeof(NetPacket) + size); | ||||||
|     packet->sender = sender; |     packet->sender = sender; | ||||||
|     packet->flags = flags; |     packet->flags = flags; | ||||||
| @@ -99,6 +106,7 @@ static void qemu_net_queue_append(NetQueue *queue, | |||||||
|     packet->sent_cb = sent_cb; |     packet->sent_cb = sent_cb; | ||||||
|     memcpy(packet->data, buf, size); |     memcpy(packet->data, buf, size); | ||||||
|  |  | ||||||
|  |     queue->nq_count++; | ||||||
|     QTAILQ_INSERT_TAIL(&queue->packets, packet, entry); |     QTAILQ_INSERT_TAIL(&queue->packets, packet, entry); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -113,6 +121,9 @@ static void qemu_net_queue_append_iov(NetQueue *queue, | |||||||
|     size_t max_len = 0; |     size_t max_len = 0; | ||||||
|     int i; |     int i; | ||||||
|  |  | ||||||
|  |     if (queue->nq_count >= queue->nq_maxlen && !sent_cb) { | ||||||
|  |         return; /* drop if queue full and no callback */ | ||||||
|  |     } | ||||||
|     for (i = 0; i < iovcnt; i++) { |     for (i = 0; i < iovcnt; i++) { | ||||||
|         max_len += iov[i].iov_len; |         max_len += iov[i].iov_len; | ||||||
|     } |     } | ||||||
| @@ -130,6 +141,7 @@ static void qemu_net_queue_append_iov(NetQueue *queue, | |||||||
|         packet->size += len; |         packet->size += len; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     queue->nq_count++; | ||||||
|     QTAILQ_INSERT_TAIL(&queue->packets, packet, entry); |     QTAILQ_INSERT_TAIL(&queue->packets, packet, entry); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -220,6 +232,7 @@ void qemu_net_queue_purge(NetQueue *queue, NetClientState *from) | |||||||
|     QTAILQ_FOREACH_SAFE(packet, &queue->packets, entry, next) { |     QTAILQ_FOREACH_SAFE(packet, &queue->packets, entry, next) { | ||||||
|         if (packet->sender == from) { |         if (packet->sender == from) { | ||||||
|             QTAILQ_REMOVE(&queue->packets, packet, entry); |             QTAILQ_REMOVE(&queue->packets, packet, entry); | ||||||
|  |             queue->nq_count--; | ||||||
|             g_free(packet); |             g_free(packet); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -233,6 +246,7 @@ bool qemu_net_queue_flush(NetQueue *queue) | |||||||
|  |  | ||||||
|         packet = QTAILQ_FIRST(&queue->packets); |         packet = QTAILQ_FIRST(&queue->packets); | ||||||
|         QTAILQ_REMOVE(&queue->packets, packet, entry); |         QTAILQ_REMOVE(&queue->packets, packet, entry); | ||||||
|  |         queue->nq_count--; | ||||||
|  |  | ||||||
|         ret = qemu_net_queue_deliver(queue, |         ret = qemu_net_queue_deliver(queue, | ||||||
|                                      packet->sender, |                                      packet->sender, | ||||||
| @@ -240,6 +254,7 @@ bool qemu_net_queue_flush(NetQueue *queue) | |||||||
|                                      packet->data, |                                      packet->data, | ||||||
|                                      packet->size); |                                      packet->size); | ||||||
|         if (ret == 0) { |         if (ret == 0) { | ||||||
|  |             queue->nq_count++; | ||||||
|             QTAILQ_INSERT_HEAD(&queue->packets, packet, entry); |             QTAILQ_INSERT_HEAD(&queue->packets, packet, entry); | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user