net_switch.cc 6.2 KB
Newer Older
Antoine Kaufmann's avatar
Antoine Kaufmann committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
/*
 * Copyright 2021 Max Planck Institute for Software Systems, and
 * National University of Singapore
 *
 * Permission is hereby granted, free of charge, to any person obtaining
 * a copy of this software and associated documentation files (the
 * "Software"), to deal in the Software without restriction, including
 * without limitation the rights to use, copy, modify, merge, publish,
 * distribute, sublicense, and/or sell copies of the Software, and to
 * permit persons to whom the Software is furnished to do so, subject to
 * the following conditions:
 *
 * The above copyright notice and this permission notice shall be
 * included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

25
26
27
28
29
#include <unistd.h>

#include <cassert>
#include <climits>
#include <csignal>
Jialin Li's avatar
Jialin Li committed
30
31
32
33
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <unordered_map>
34
#include <vector>
Jialin Li's avatar
Jialin Li committed
35
36

extern "C" {
37
#include <simbricks/netif/netif.h>
38
#include <simbricks/proto/base.h>
Jialin Li's avatar
Jialin Li committed
39
40
};

Antoine Kaufmann's avatar
Antoine Kaufmann committed
41
42
static uint64_t sync_period = (500 * 1000ULL);  // 500ns
static uint64_t eth_latency = (500 * 1000ULL);  // 500ns
Jialin Li's avatar
Jialin Li committed
43
44
45

/* MAC address type */
struct MAC {
46
  const volatile uint8_t *data;
Jialin Li's avatar
Jialin Li committed
47

48
  explicit MAC(const volatile uint8_t *data) : data(data) {
49
  }
Jialin Li's avatar
Jialin Li committed
50

51
52
53
54
55
  bool operator==(const MAC &other) const {
    for (int i = 0; i < 6; i++) {
      if (data[i] != other.data[i]) {
        return false;
      }
Jialin Li's avatar
Jialin Li committed
56
    }
57
58
    return true;
  }
Jialin Li's avatar
Jialin Li committed
59
60
};
namespace std {
Antoine Kaufmann's avatar
Antoine Kaufmann committed
61
template <>
62
63
struct hash<MAC> {
  size_t operator()(const MAC &m) const {
Antoine Kaufmann's avatar
Antoine Kaufmann committed
64
65
    size_t res = 0;
    for (int i = 0; i < 6; i++) {
66
      res = (res << 4) | (res ^ m.data[i]);
Antoine Kaufmann's avatar
Antoine Kaufmann committed
67
68
    }
    return res;
69
  }
Antoine Kaufmann's avatar
Antoine Kaufmann committed
70
71
};
}  // namespace std
Jialin Li's avatar
Jialin Li committed
72
73
74
75
76
77

/* Global variables */
static uint64_t cur_ts = 0;
static int exiting = 0;
static const volatile uint8_t bcast[6] = {0xFF};
static const MAC bcast_addr(bcast);
78
static std::vector<struct SimbricksNetIf> nsifs;
Jialin Li's avatar
Jialin Li committed
79
80
static std::unordered_map<MAC, int> mac_table;

81
82
static void sigint_handler(int dummy) {
  exiting = 1;
Jialin Li's avatar
Jialin Li committed
83
84
}

85
static void forward_pkt(volatile struct SimbricksProtoNetD2NSend *tx,
86
                        int port) {
87
  volatile union SimbricksProtoNetN2D *msg_to;
88
  msg_to = SimbricksNetIfN2DAlloc(&nsifs[port], cur_ts, eth_latency);
89
  if (msg_to != NULL) {
90
    volatile struct SimbricksProtoNetN2DRecv *rx;
91
92
93
94
95
96
    rx = &msg_to->recv;
    rx->len = tx->len;
    rx->port = 0;
    memcpy((void *)rx->data, (void *)tx->data, tx->len);

    // WMB();
97
98
    rx->own_type = SIMBRICKS_PROTO_NET_N2D_MSG_RECV |
        SIMBRICKS_PROTO_NET_N2D_OWN_DEV;
99
100
101
  } else {
    fprintf(stderr, "forward_pkt: dropping packet\n");
  }
Jialin Li's avatar
Jialin Li committed
102
103
}

104
static void switch_pkt(struct SimbricksNetIf *nsif, int iport) {
105
  volatile union SimbricksProtoNetD2N *msg_from =
106
      SimbricksNetIfD2NPoll(nsif, cur_ts);
107
108
109
110
  if (msg_from == NULL) {
    return;
  }

111
112
113
  uint8_t type = msg_from->dummy.own_type & SIMBRICKS_PROTO_NET_D2N_MSG_MASK;
  if (type == SIMBRICKS_PROTO_NET_D2N_MSG_SEND) {
    volatile struct SimbricksProtoNetD2NSend *tx;
114
115
116
117
118
119
    tx = &msg_from->send;
    // Get MAC addresses
    MAC dst(tx->data), src(tx->data + 6);
    // MAC learning
    if (!(src == bcast_addr)) {
      mac_table[src] = iport;
Jialin Li's avatar
Jialin Li committed
120
    }
121
122
123
124
    // L2 forwarding
    if (mac_table.count(dst) > 0) {
      int eport = mac_table.at(dst);
      forward_pkt(tx, eport);
Jialin Li's avatar
Jialin Li committed
125
    } else {
126
127
128
129
130
131
132
      // Broadcast
      for (int eport = 0; eport < nsifs.size(); eport++) {
        if (eport != iport) {
          // Do not forward to ingress port
          forward_pkt(tx, eport);
        }
      }
Jialin Li's avatar
Jialin Li committed
133
    }
134
  } else if (type == SIMBRICKS_PROTO_NET_D2N_MSG_SYNC) {
135
136
137
138
  } else {
    fprintf(stderr, "switch_pkt: unsupported type=%u\n", type);
    abort();
  }
139
  SimbricksNetIfD2NDone(nsif, msg_from);
Jialin Li's avatar
Jialin Li committed
140
141
}

142
143
144
int main(int argc, char *argv[]) {
  int c;
  int bad_option = 0;
145
  int sync_mode = SIMBRICKS_PROTO_SYNC_SIMBRICKS;
146
147
148
149
150

  // Parse command line argument
  while ((c = getopt(argc, argv, "s:S:E:m:")) != -1 && !bad_option) {
    switch (c) {
      case 's': {
151
        struct SimbricksNetIf nsif;
152
        int sync = 1;
153
        if (SimbricksNetIfInit(&nsif, optarg, &sync) != 0) {
154
155
          fprintf(stderr, "connecting to %s failed\n", optarg);
          return EXIT_FAILURE;
Jialin Li's avatar
Jialin Li committed
156
        }
157
158
159
160
161
162
163
164
165
166
167
168
169
170
        nsifs.push_back(nsif);
        break;
      }

      case 'S':
        sync_period = strtoull(optarg, NULL, 0) * 1000ULL;
        break;

      case 'E':
        eth_latency = strtoull(optarg, NULL, 0) * 1000ULL;
        break;

      case 'm':
        sync_mode = strtol(optarg, NULL, 0);
171
172
        assert(sync_mode == SIMBRICKS_PROTO_SYNC_SIMBRICKS ||
            sync_mode == SIMBRICKS_PROTO_SYNC_BARRIER);
173
174
175
176
177
178
        break;

      default:
        fprintf(stderr, "unknown option %c\n", c);
        bad_option = 1;
        break;
Jialin Li's avatar
Jialin Li committed
179
    }
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
  }

  if (nsifs.empty() || bad_option) {
    fprintf(stderr,
            "Usage: net_switch [-S SYNC-PERIOD] [-E ETH-LATENCY] "
            "-s SOCKET-A [-s SOCKET-B ...]\n");
    return EXIT_FAILURE;
  }

  signal(SIGINT, sigint_handler);
  signal(SIGTERM, sigint_handler);

  printf("start polling\n");
  while (!exiting) {
    // Sync all interfaces
    for (auto &nsif : nsifs) {
196
197
198
      if (SimbricksNetIfN2DSync(&nsif, cur_ts, eth_latency, sync_period,
          sync_mode) != 0) {
        fprintf(stderr, "SimbricksNetIfN2DSync failed\n");
199
200
        abort();
      }
Jialin Li's avatar
Jialin Li committed
201
    }
202
    SimbricksNetIfAdvanceEpoch(cur_ts, sync_period, sync_mode);
203
204
205
206
207
208
209
210
211

    // Switch packets
    uint64_t min_ts;
    do {
      min_ts = ULLONG_MAX;
      for (int port = 0; port < nsifs.size(); port++) {
        auto &nsif = nsifs.at(port);
        switch_pkt(&nsif, port);
        if (nsif.sync) {
212
          uint64_t ts = SimbricksNetIfD2NTimestamp(&nsif);
213
          min_ts = ts < min_ts ? ts : min_ts;
Jialin Li's avatar
Jialin Li committed
214
        }
215
216
217
218
219
      }
    } while (!exiting && (min_ts <= cur_ts));

    // Update cur_ts
    if (min_ts < ULLONG_MAX) {
220
      cur_ts = SimbricksNetIfAdvanceTime(min_ts, sync_period, sync_mode);
Jialin Li's avatar
Jialin Li committed
221
    }
222
  }
Jialin Li's avatar
Jialin Li committed
223

224
  return 0;
Jialin Li's avatar
Jialin Li committed
225
}