summaryrefslogtreecommitdiffstats
path: root/src/svm/queue.h
blob: 9d21b24d7ea258650443a75e7c25163037d5f37f (plain)
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
/*
 *------------------------------------------------------------------
 * svm_queue.h - shared-memory queues
 *
 * Copyright (c) 2009-2019 Cisco and/or its affiliates.
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at:
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *------------------------------------------------------------------
 */

#ifndef included_svm_queue_h
#define included_svm_queue_h

#include <pthread.h>

typedef struct _svm_queue
{
  pthread_mutex_t mutex;	/* 8 bytes */
  pthread_cond_t condvar;	/* 8 bytes */
  int head;
  int tail;
  volatile int cursize;
  int maxsize;
  int elsize;
  int consumer_pid;
  int producer_evtfd;
  int consumer_evtfd;
  char data[0];
} svm_queue_t;

typedef enum
{
  SVM_Q_WAIT = 0,	/**< blocking call - best used in combination with
			     condvars, for eventfds we don't yield the cpu */
  SVM_Q_NOWAIT,		/**< non-blocking call - works with both condvar and
			     eventfd signaling */
  SVM_Q_TIMEDWAIT,	/**< blocking call, returns on signal or time-out -
			     best used in combination with condvars, with
			     eventfds we don't yield the cpu */
} svm_q_conditional_wait_t;

/**
 * Allocate and initialize svm queue
 *
 * @param nels		number of elements on the queue
 * @param elsize	element size, presumably 4 and cacheline-size will
 *          		be popular choices.
 * @param pid   	consumer pid
 * @return 		a newly initialized svm queue
 *
 * The idea is to call this function in the queue consumer,
 * and e-mail the queue pointer to the producer(s).
 *
 * The vpp process / main thread allocates one of these
 * at startup; its main input queue. The vpp main input queue
 * has a pointer to it in the shared memory segment header.
 *
 * You probably want to be on an svm data heap before calling this
 * function.
 */
svm_queue_t *svm_queue_alloc_and_init (int nels, int elsize,
				       int consumer_pid);
svm_queue_t *svm_queue_init (void *base, int nels, int elsize);
void svm_queue_free (svm_queue_t * q);
int svm_queue_add (svm_queue_t * q, u8 * elem, int nowait);
int svm_queue_add2 (svm_queue_t * q, u8 * elem, u8 * elem2, int nowait);
int svm_queue_sub (svm_queue_t * q, u8 * elem, svm_q_conditional_wait_t cond,
		   u32 time);
int svm_queue_sub2 (svm_queue_t * q, u8 * elem);
void svm_queue_lock (svm_queue_t * q);
void svm_queue_send_signal (svm_queue_t * q, u8 is_prod);
void svm_queue_unlock (svm_queue_t * q);
int svm_queue_is_full (svm_queue_t * q);
int svm_queue_add_nolock (svm_queue_t * q, u8 * elem);
int svm_queue_sub_raw (svm_queue_t * q, u8 * elem);

/**
 * Wait for queue event
 *
 * Must be called with mutex held.
 */
void svm_queue_wait (svm_queue_t * q);

/**
 * Timed wait for queue event
 *
 * Must be called with mutex held.
 *
 * @param q		svm queue
 * @param timeout	time in seconds
 * @return 		0 on success, ETIMEDOUT on timeout or an error
 */
int svm_queue_timedwait (svm_queue_t * q, double timeout);

/**
 * Add element to queue with mutex held
 * @param q		queue
 * @param elem		pointer element data to add
 */
void svm_queue_add_raw (svm_queue_t * q, u8 * elem);

/**
 * Set producer's event fd
 *
 * When the producer must generate an event it writes 1 to the provided fd.
 * Once this is set, condvars are not used anymore for signaling.
 */
void svm_queue_set_producer_event_fd (svm_queue_t * q, int fd);

/**
 * Set consumer's event fd
 *
 * When the consumer must generate an event it writes 1 to the provided fd.
 * Although in practice the two fds point to the same underlying file
 * description, because the producer and consumer are different processes
 * the descriptors will be different. It's the caller's responsibility to
 * ensure the file descriptors are properly exchanged between the two peers.
 */
void svm_queue_set_consumer_event_fd (svm_queue_t * q, int fd);

/*
 * DEPRECATED please use svm_queue_t instead
 */
typedef svm_queue_t unix_shared_memory_queue_t;

#endif /* included_svm_queue_h */

/*
 * fd.io coding-style-patch-verification: ON
 *
 * Local Variables:
 * eval: (c-set-style "gnu")
 * End:
 */
return s; } /* Ascii buffer and length. */ u8 * format_ascii_bytes (u8 * s, va_list * va) { u8 *v = va_arg (*va, u8 *); uword n_bytes = va_arg (*va, uword); vec_add (s, v, n_bytes); return s; } /* Format hex dump. */ u8 * format_hex_bytes (u8 * s, va_list * va) { u8 *bytes = va_arg (*va, u8 *); int n_bytes = va_arg (*va, int); uword i; /* Print short or long form depending on byte count. */ uword short_form = n_bytes <= 32; u32 indent = format_get_indent (s); if (n_bytes == 0) return s; for (i = 0; i < n_bytes; i++) { if (!short_form && (i % 32) == 0) s = format (s, "%08x: ", i); s = format (s, "%02x", bytes[i]); if (!short_form && ((i + 1) % 32) == 0 && (i + 1) < n_bytes) s = format (s, "\n%U", format_white_space, indent); } return s; } /* Add variable number of spaces. */ u8 * format_white_space (u8 * s, va_list * va) { u32 n = va_arg (*va, u32); while (n-- > 0) vec_add1 (s, ' '); return s; } u8 * format_time_interval (u8 * s, va_list * args) { u8 *fmt = va_arg (*args, u8 *); f64 t = va_arg (*args, f64); u8 *f; const f64 seconds_per_minute = 60; const f64 seconds_per_hour = 60 * seconds_per_minute; const f64 seconds_per_day = 24 * seconds_per_hour; uword days, hours, minutes, secs, msecs, usecs; days = t / seconds_per_day; t -= days * seconds_per_day; hours = t / seconds_per_hour; t -= hours * seconds_per_hour; minutes = t / seconds_per_minute; t -= minutes * seconds_per_minute; secs = t; t -= secs; msecs = 1e3 * t; usecs = 1e6 * t; for (f = fmt; *f; f++) { uword what, c; char *what_fmt = "%d"; switch (c = *f) { default: vec_add1 (s, c); continue; case 'd': what = days; what_fmt = "%d"; break; case 'h': what = hours; what_fmt = "%02d"; break; case 'm': what = minutes; what_fmt = "%02d"; break; case 's': what = secs; what_fmt = "%02d"; break; case 'f': what = msecs; what_fmt = "%03d"; break; case 'u': what = usecs; what_fmt = "%06d"; break; } s = format (s, what_fmt, what); } return s; } /* Unparse memory size e.g. 100, 100k, 100m, 100g. */ u8 * format_memory_size (u8 * s, va_list * va) { uword size = va_arg (*va, uword); uword l, u, log_u; l = size > 0 ? min_log2 (size) : 0; if (l < 10) log_u = 0; else if (l < 20) log_u = 10; else if (l < 30) log_u = 20; else log_u = 30; u = (uword) 1 << log_u; if (size & (u - 1)) s = format (s, "%.2f", (f64) size / (f64) u); else s = format (s, "%d", size >> log_u); if (log_u != 0) s = format (s, "%c", " kmg"[log_u / 10]); return s; } /* Parse memory size e.g. 100, 100k, 100m, 100g. */ uword unformat_memory_size (unformat_input_t * input, va_list * va) { uword amount, shift, c; uword *result = va_arg (*va, uword *); if (!unformat (input, "%wd%_", &amount)) return 0; c = unformat_get_input (input); switch (c) { case 'k': case 'K': shift = 10; break; case 'm': case 'M': shift = 20; break; case 'g': case 'G': shift = 30; break; default: shift = 0; unformat_put_input (input); break; } *result = amount << shift; return 1; } /* Format c identifier: e.g. a_name -> "a name". Words for both vector names and null terminated c strings. */ u8 * format_c_identifier (u8 * s, va_list * va) { u8 *id = va_arg (*va, u8 *); uword i, l; l = ~0; if (clib_mem_is_vec (id)) l = vec_len (id); if (id) for (i = 0; id[i] != 0 && i < l; i++) { u8 c = id[i]; if (c == '_') c = ' '; vec_add1 (s, c); } return s; } u8 * format_hexdump (u8 * s, va_list * args) { u8 *data = va_arg (*args, u8 *); uword len = va_arg (*args, uword); int i, index = 0; const int line_len = 16; u8 *line_hex = 0; u8 *line_str = 0; u32 indent = format_get_indent (s); if (!len) return s; for (i = 0; i < len; i++) { line_hex = format (line_hex, "%02x ", data[i]); line_str = format (line_str, "%c", isprint (data[i]) ? data[i] : '.'); if (!((i + 1) % line_len)) { s = format (s, "%U%05x: %v[%v]", format_white_space, index ? indent : 0, index, line_hex, line_str); if (i < len - 1) s = format (s, "\n"); index = i + 1; vec_reset_length (line_hex); vec_reset_length (line_str); } } while (i++ % line_len) line_hex = format (line_hex, " "); if (vec_len (line_hex)) s = format (s, "%U%05x: %v[%v]", format_white_space, indent, index, line_hex, line_str); vec_free (line_hex); vec_free (line_str); return s; } /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */