scc  2024.06
SystemC components library
target_mixin.h
1 /*******************************************************************************
2  * Copyright 2016, 2018 MINRES Technologies GmbH
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  *******************************************************************************/
16 #ifndef _TLM_NW_TARGET_MIXIN_H_
17 #define _TLM_NW_TARGET_MIXIN_H_
18 
19 #ifndef SC_INCLUDE_DYNAMIC_PROCESSES // needed for sc_spawn
20 #define SC_INCLUDE_DYNAMIC_PROCESSES
21 #endif
22 
23 #include "scc/utilities.h"
24 #include <functional>
25 #include <sstream>
26 #include <tlm>
27 #include <tlm_utils/peq_with_get.h>
28 
30 namespace tlm {
32 namespace nw {
36 template <typename BASE_TYPE, typename TYPES = typename BASE_TYPE::protocol_types> class target_mixin : public BASE_TYPE {
37  // friend class fw_process;
38  // friend class bw_process;
39 
40 public:
41  using transaction_type = typename TYPES::tlm_payload_type;
42  using phase_type = typename TYPES::tlm_phase_type;
43  using sync_enum_type = tlm::tlm_sync_enum;
44  using fw_interface_type = tlm::tlm_fw_transport_if<TYPES>;
45  using bw_interface_type = tlm::tlm_bw_transport_if<TYPES>;
46 
47 public:
52  : target_mixin(sc_core::sc_gen_unique_name("target_mixin_socket")) {}
58  explicit target_mixin(const sc_core::sc_module_name& n)
59  : BASE_TYPE(n)
60  , m_fw_process(this)
61  , m_bw_process(this)
62  , m_current_transaction(nullptr) {
63  bind(m_fw_process);
64  }
66  using BASE_TYPE::bind;
72  tlm::tlm_bw_transport_if<TYPES>* operator->() { return &m_bw_process; }
78  void register_nb_transport_fw(std::function<sync_enum_type(transaction_type&, phase_type&, sc_core::sc_time&)> cb) {
79  assert(!sc_core::sc_get_curr_simcontext()->elaboration_done());
80  m_fw_process.set_nb_transport_ptr(cb);
81  }
87  void register_b_transport(std::function<void(transaction_type&, sc_core::sc_time&)> cb) {
88  assert(!sc_core::sc_get_curr_simcontext()->elaboration_done());
89  m_fw_process.set_b_transport_ptr(cb);
90  }
95  void register_transport_dbg(std::function<unsigned int(transaction_type&)> cb) {
96  assert(!sc_core::sc_get_curr_simcontext()->elaboration_done());
97  m_fw_process.set_transport_dbg_ptr(cb);
98  }
104  void register_get_direct_mem_ptr(std::function<bool(transaction_type&, tlm::tlm_dmi&)> cb) {
105  assert(!sc_core::sc_get_curr_simcontext()->elaboration_done());
106  m_fw_process.set_get_direct_mem_ptr(cb);
107  }
108 
109 private:
110  // make call on bw path.
111  sync_enum_type bw_nb_transport(transaction_type& trans, phase_type& phase, sc_core::sc_time& t) {
112  return BASE_TYPE::operator->()->nb_transport_bw(trans, phase, t);
113  }
114 
115  // Helper class to handle bw path calls
116  // Needed to detect transaction end when called from b_transport.
117  class bw_process : public tlm::nw::tlm_network_bw_transport_if<TYPES> {
118  public:
119  bw_process(target_mixin* p_own)
120  : m_owner(p_own) {}
121 
122  sync_enum_type nb_transport_bw(transaction_type& trans, phase_type& phase, sc_core::sc_time& t) {
123  typename std::map<transaction_type*, sc_core::sc_event*>::iterator it;
124 
125  it = m_owner->m_pending_trans.find(&trans);
126  if(it == m_owner->m_pending_trans.end()) {
127  // Not a blocking call, forward.
128  return m_owner->bw_nb_transport(trans, phase, t);
129 
130  } else {
131  if(phase == tlm::END_REQ) {
132  m_owner->m_end_request.notify(sc_core::SC_ZERO_TIME);
133  return tlm::TLM_ACCEPTED;
134 
135  } else if(phase == tlm::BEGIN_RESP) {
136  if(m_owner->m_current_transaction == &trans) {
137  m_owner->m_end_request.notify(sc_core::SC_ZERO_TIME);
138  }
139  // TODO: add response-accept delay?
140  it->second->notify(t);
141  m_owner->m_pending_trans.erase(it);
142  return tlm::TLM_COMPLETED;
143 
144  } else {
145  assert(false);
146  exit(1);
147  }
148 
149  // return tlm::TLM_COMPLETED; //Should not reach here
150  }
151  }
152 
153  private:
154  target_mixin* m_owner;
155  };
156 
157  class fw_process : public tlm::nw::tlm_network_fw_transport_if<TYPES>, public tlm::nw::tlm_base_mm_interface {
158  public:
159  using NBTransportPtr = std::function<sync_enum_type(transaction_type&, phase_type&, sc_core::sc_time&)>;
160  using BTransportPtr = std::function<void(transaction_type&, sc_core::sc_time&)>;
161  using TransportDbgPtr = std::function<unsigned int(transaction_type&)>;
162  using GetDirectMemPtr = std::function<bool(transaction_type&, tlm::tlm_dmi&)>;
163 
164  fw_process(target_mixin* p_own)
165  : m_name(p_own->name())
166  , m_owner(p_own)
167  , m_nb_transport_ptr(0)
168  , m_b_transport_ptr(0)
169  , m_transport_dbg_ptr(0)
170  , m_get_direct_mem_ptr(0)
171  , m_peq(sc_core::sc_gen_unique_name("m_peq"))
172  , m_response_in_progress(false) {
173  sc_core::sc_spawn_options opts;
174  opts.set_sensitivity(&m_peq.get_event());
175  sc_core::sc_spawn(sc_bind(&fw_process::b2nb_thread, this), sc_core::sc_gen_unique_name("b2nb_thread"), &opts);
176  }
177 
178  void set_nb_transport_ptr(NBTransportPtr p) {
179  if(m_nb_transport_ptr) {
180  std::stringstream s;
181  s << m_name << ": non-blocking callback allready registered";
182  SC_REPORT_WARNING("/OSCI_TLM-2/simple_socket", s.str().c_str());
183  } else {
184  m_nb_transport_ptr = p;
185  }
186  }
187 
188  void set_b_transport_ptr(BTransportPtr p) {
189  if(m_b_transport_ptr) {
190  std::stringstream s;
191  s << m_name << ": blocking callback allready registered";
192  SC_REPORT_WARNING("/OSCI_TLM-2/simple_socket", s.str().c_str());
193  } else {
194  m_b_transport_ptr = p;
195  }
196  }
197 
198  void set_transport_dbg_ptr(TransportDbgPtr p) {
199  if(m_transport_dbg_ptr) {
200  std::stringstream s;
201  s << m_name << ": debug callback allready registered";
202  SC_REPORT_WARNING("/OSCI_TLM-2/simple_socket", s.str().c_str());
203  } else {
204  m_transport_dbg_ptr = p;
205  }
206  }
207 
208  void set_get_direct_mem_ptr(GetDirectMemPtr p) {
209  if(m_get_direct_mem_ptr) {
210  std::stringstream s;
211  s << m_name << ": get DMI pointer callback allready registered";
212  SC_REPORT_WARNING("/OSCI_TLM-2/simple_socket", s.str().c_str());
213  } else {
214  m_get_direct_mem_ptr = p;
215  }
216  }
217  // Interface implementation
218  sync_enum_type nb_transport_fw(transaction_type& trans, phase_type& phase, sc_core::sc_time& t) {
219  if(m_nb_transport_ptr) {
220  // forward call
221  return m_nb_transport_ptr(trans, phase, t);
222  } else if(m_b_transport_ptr) {
223  if(phase == tlm::BEGIN_REQ) {
224  // prepare thread to do blocking call
225  process_handle_class* ph = m_process_handle.get_handle(&trans);
226 
227  if(!ph) { // create new dynamic process
228  ph = new process_handle_class(&trans);
229  m_process_handle.put_handle(ph);
230 
231  sc_core::sc_spawn_options opts;
232  opts.dont_initialize();
233  opts.set_sensitivity(&ph->m_e);
234  sc_core::sc_spawn(sc_bind(&fw_process::nb2b_thread, this, ph), sc_core::sc_gen_unique_name("nb2b_thread"), &opts);
235  }
236  ph->m_e.notify(t);
237  phase = tlm::END_REQ;
238  return tlm::TLM_UPDATED;
239  } else if(phase == tlm::END_RESP) {
240  m_response_in_progress = false;
241  m_end_response.notify(t);
242  return tlm::TLM_COMPLETED;
243  } else {
244  assert(false);
245  exit(1);
246  }
247  } else {
248  std::stringstream s;
249  s << m_name << ": no non-blocking transport callback registered";
250  SC_REPORT_ERROR("/OSCI_TLM-2/simple_socket", s.str().c_str());
251  }
252  return tlm::TLM_ACCEPTED;
253  }
254 
255  void b_transport(transaction_type& trans, sc_core::sc_time& t) {
256  if(m_b_transport_ptr) {
257  // forward call
258  m_b_transport_ptr(trans, t);
259  return;
260 
261  } else if(m_nb_transport_ptr) {
262  m_peq.notify(trans, t);
263  t = sc_core::SC_ZERO_TIME;
264 
265  mm_end_event_ext mm_ext;
266  const bool mm_added = !trans.has_mm();
267 
268  if(mm_added) {
269  trans.set_mm(this);
270  trans.set_auto_extension(&mm_ext);
271  trans.acquire();
272  }
273 
274  // wait until transaction is finished
275  sc_core::sc_event end_event;
276  m_owner->m_pending_trans[&trans] = &end_event;
277  sc_core::wait(end_event);
278 
279  if(mm_added) {
280  // release will not delete the transaction, it will notify mm_ext.done
281  trans.release();
282  if(trans.get_ref_count()) {
283  sc_core::wait(mm_ext.done);
284  }
285  trans.set_mm(0);
286  }
287 
288  } else {
289  std::stringstream s;
290  s << m_name << ": no blocking transport callback registered";
291  SC_REPORT_ERROR("/OSCI_TLM-2/simple_socket", s.str().c_str());
292  }
293  }
294 
295  unsigned int transport_dbg(transaction_type& trans) {
296  if(m_transport_dbg_ptr) {
297  // forward call
298  return m_transport_dbg_ptr(trans);
299  } else {
300  // No debug support
301  return 0;
302  }
303  }
304 
305  bool get_direct_mem_ptr(transaction_type& trans, tlm::tlm_dmi& dmi_data) {
306  if(m_get_direct_mem_ptr) {
307  // forward call
308  return m_get_direct_mem_ptr(trans, dmi_data);
309 
310  } else {
311  // No DMI support
312  dmi_data.allow_read_write();
313  dmi_data.set_start_address(0x0);
314  dmi_data.set_end_address((sc_dt::uint64)-1);
315  return false;
316  }
317  }
318 
319  private:
320  // dynamic process handler for nb2b conversion
321 
322  class process_handle_class {
323  public:
324  explicit process_handle_class(transaction_type* trans)
325  : m_trans(trans)
326  , m_suspend(false) {}
327 
328  transaction_type* m_trans;
329  sc_core::sc_event m_e;
330  bool m_suspend;
331  };
332 
333  class process_handle_list {
334  public:
335  process_handle_list() = default;
336 
337  ~process_handle_list() {
338  for(typename std::vector<process_handle_class*>::iterator it = v.begin(), end = v.end(); it != end; ++it)
339  delete *it;
340  }
341 
342  process_handle_class* get_handle(transaction_type* trans) {
343  typename std::vector<process_handle_class*>::iterator it;
344 
345  for(it = v.begin(); it != v.end(); it++) {
346  if((*it)->m_suspend) { // found suspended dynamic process, re-use it
347  (*it)->m_trans = trans; // replace to new one
348  (*it)->m_suspend = false;
349  return *it;
350  }
351  }
352  return NULL; // no suspended process
353  }
354 
355  void put_handle(process_handle_class* ph) { v.push_back(ph); }
356 
357  private:
358  std::vector<process_handle_class*> v;
359  };
360 
361  process_handle_list m_process_handle;
362 
363  void nb2b_thread(process_handle_class* h) {
364 
365  while(true) {
366  transaction_type* trans = h->m_trans;
367  sc_core::sc_time t = sc_core::SC_ZERO_TIME;
368 
369  // forward call
370  m_b_transport_ptr(*trans, t);
371 
372  sc_core::wait(t);
373 
374  // return path
375  while(m_response_in_progress) {
376  sc_core::wait(m_end_response);
377  }
378  t = sc_core::SC_ZERO_TIME;
379  phase_type phase = tlm::BEGIN_RESP;
380  sync_enum_type sync = m_owner->bw_nb_transport(*trans, phase, t);
381  if(!(sync == tlm::TLM_COMPLETED || (sync == tlm::TLM_UPDATED && phase == tlm::END_RESP))) {
382  m_response_in_progress = true;
383  }
384 
385  // suspend until next transaction
386  h->m_suspend = true;
387  sc_core::wait();
388  }
389  }
390 
391  void b2nb_thread() {
392  while(true) {
393  sc_core::wait(m_peq.get_event());
394 
395  transaction_type* trans;
396  while((trans = m_peq.get_next_transaction()) != 0) {
397  assert(m_nb_transport_ptr);
398  phase_type phase = tlm::BEGIN_REQ;
399  sc_core::sc_time t = sc_core::SC_ZERO_TIME;
400 
401  switch(m_nb_transport_ptr(*trans, phase, t)) {
402  case tlm::TLM_COMPLETED: {
403  // notify transaction is finished
404  typename std::map<transaction_type*, sc_core::sc_event*>::iterator it = m_owner->m_pending_trans.find(trans);
405  assert(it != m_owner->m_pending_trans.end());
406  it->second->notify(t);
407  m_owner->m_pending_trans.erase(it);
408  break;
409  }
410 
411  case tlm::TLM_ACCEPTED:
412  case tlm::TLM_UPDATED:
413  switch(phase) {
414  case tlm::BEGIN_REQ:
415  m_owner->m_current_transaction = trans;
416  sc_core::wait(m_owner->m_end_request);
417  m_owner->m_current_transaction = 0;
418  break;
419 
420  case tlm::END_REQ:
421  sc_core::wait(t);
422  break;
423 
424  case tlm::BEGIN_RESP: {
425  phase = tlm::END_RESP;
426  sc_core::wait(t); // This line is a bug fix added in TLM-2.0.2
427  t = sc_core::SC_ZERO_TIME;
428  m_nb_transport_ptr(*trans, phase, t);
429 
430  // notify transaction is finished
431  typename std::map<transaction_type*, sc_core::sc_event*>::iterator it = m_owner->m_pending_trans.find(trans);
432  assert(it != m_owner->m_pending_trans.end());
433  it->second->notify(t);
434  m_owner->m_pending_trans.erase(it);
435  break;
436  }
437 
438  default:
439  assert(false);
440  exit(1);
441  };
442  break;
443 
444  default:
445  assert(false);
446  exit(1);
447  };
448  }
449  }
450  }
451 
452  void free(tlm::nw::tlm_network_payload_base* trans) {
453  mm_end_event_ext* ext = trans->template get_extension<mm_end_event_ext>();
454  assert(ext);
455  // notif event first before freeing extensions (reset)
456  ext->done.notify();
457  trans->reset();
458  }
459 
460  private:
461  class mm_end_event_ext : public tlm::tlm_extension<mm_end_event_ext> {
462  public:
463  tlm::tlm_extension_base* clone() const { return NULL; }
464  void free() {}
465  void copy_from(tlm::tlm_extension_base const&) {}
466  sc_core::sc_event done;
467  };
468 
469  private:
470  const std::string m_name;
471  target_mixin* m_owner;
472  NBTransportPtr m_nb_transport_ptr;
473  BTransportPtr m_b_transport_ptr;
474  TransportDbgPtr m_transport_dbg_ptr;
475  GetDirectMemPtr m_get_direct_mem_ptr;
476  tlm_utils::peq_with_get<transaction_type> m_peq;
477  bool m_response_in_progress;
478  sc_core::sc_event m_end_response;
479  };
480 
481 private:
482  fw_process m_fw_process;
483  bw_process m_bw_process;
484  std::map<transaction_type*, sc_core::sc_event*> m_pending_trans;
485  sc_core::sc_event m_end_request;
486  transaction_type* m_current_transaction;
487 };
488 } // namespace nw
489 } // namespace tlm
490 
491 #endif //_TLM_NW_TARGET_MIXIN_H_
tlm::tlm_bw_transport_if< TYPES > * operator->()
Definition: target_mixin.h:72
void register_transport_dbg(std::function< unsigned int(transaction_type &)> cb)
Definition: target_mixin.h:95
void register_b_transport(std::function< void(transaction_type &, sc_core::sc_time &)> cb)
Definition: target_mixin.h:87
void register_get_direct_mem_ptr(std::function< bool(transaction_type &, tlm::tlm_dmi &)> cb)
Definition: target_mixin.h:104
target_mixin(const sc_core::sc_module_name &n)
Definition: target_mixin.h:58
void register_nb_transport_fw(std::function< sync_enum_type(transaction_type &, phase_type &, sc_core::sc_time &)> cb)
Definition: target_mixin.h:78
SystemC TLM.
Definition: cxs_tlm.h:69