You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

FunctionTest.cs 24 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  1. using Microsoft.VisualStudio.TestTools.UnitTesting;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using static Tensorflow.Native.UnitTest.c_test_util;
  6. namespace Tensorflow.Native.UnitTest
  7. {
  8. /// <summary>
  9. /// tensorflow\c\c_api_function_test.cc
  10. /// `class CApiColocationTest`
  11. /// </summary>
  12. [TestClass]
  13. public class FunctionTest : CApiTest, IDisposable
  14. {
  15. Graph func_graph_;
  16. Graph host_graph_;
  17. string func_name_ = "MyFunc";
  18. string func_node_name_ = "MyFunc_0";
  19. Status s_;
  20. IntPtr func_;
  21. [TestInitialize]
  22. public void Initialize()
  23. {
  24. func_graph_ = new Graph();
  25. host_graph_ = new Graph();
  26. s_ = new Status();
  27. }
  28. [TestMethod]
  29. public void OneOp_ZeroInputs_OneOutput()
  30. {
  31. var c = ScalarConst(10, func_graph_, s_, "scalar10");
  32. // Define
  33. Define(-1, new Operation[0], new Operation[0], new[] { c }, new string[0]);
  34. // Use, run, and verify
  35. var func_op = Use(new Operation[0]);
  36. Run(new KeyValuePair<Operation, Tensor>[0], func_op, 10);
  37. VerifyFDef(new[] { "scalar10_0" },
  38. new List<IOSpec>(),
  39. new List<IOSpec> { new IOSpec("scalar10", DataType.DtInt32) },
  40. new List<EdgeSpec> { new EdgeSpec("scalar10_0:output:0", "scalar10") },
  41. new List<EdgeSpec>());
  42. }
  43. [TestMethod]
  44. public void OneOp_OneInput_OneOutput()
  45. {
  46. // Define
  47. var feed = Placeholder(func_graph_, s_);
  48. var neg = Neg(feed, func_graph_, s_);
  49. Define(-1, new Operation[0], new[] { feed }, new[] { neg }, new string[0]);
  50. // Use, run, and verify
  51. var func_feed = Placeholder(host_graph_, s_);
  52. var func_op = Use(new[] { func_feed });
  53. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) }, func_op, -3);
  54. VerifyFDef(new string[] { "neg_0" },
  55. new List<IOSpec> { new IOSpec("feed", DataType.DtInt32) },
  56. new List<IOSpec> { new IOSpec("neg", DataType.DtInt32) },
  57. new List<EdgeSpec> { new EdgeSpec("feed", "neg_0:0"), new EdgeSpec("neg_0:y:0", "neg") },
  58. new List<EdgeSpec>());
  59. }
  60. [TestMethod]
  61. public void OneOutput_OutputNames()
  62. {
  63. // Define
  64. var feed = Placeholder(func_graph_, s_);
  65. var neg = Neg(feed, func_graph_, s_);
  66. Define(-1,
  67. new Operation[0],
  68. new[] { feed },
  69. new[] { neg },
  70. new[] { "negated_num" });
  71. // Use, run, and verify
  72. var func_feed = Placeholder(host_graph_, s_);
  73. var func_op = Use(new[] { func_feed });
  74. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) }, func_op, -3);
  75. VerifyFDef(new string[] { "neg" },
  76. new List<IOSpec> { new IOSpec("feed", DataType.DtInt32) },
  77. new List<IOSpec> { new IOSpec("negated_num", DataType.DtInt32) },
  78. new List<EdgeSpec> { new EdgeSpec("feed", "neg:0"), new EdgeSpec("neg:y:0", "negated_num") },
  79. new List<EdgeSpec>());
  80. }
  81. [TestMethod]
  82. public void OutputNames_SameNameAsInput()
  83. {
  84. // Define
  85. var feed = Placeholder(func_graph_, s_, "negation");
  86. var neg = Neg(feed, func_graph_, s_, "neg");
  87. Define(-1,
  88. new Operation[0],
  89. new[] { feed },
  90. new[] { neg },
  91. new[] { "negation" });
  92. // Use, run, and verify
  93. var func_feed = Placeholder(host_graph_, s_);
  94. var func_op = Use(new[] { func_feed });
  95. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) }, func_op, -3);
  96. VerifyFDef(new string[] { "neg" },
  97. new List<IOSpec> { new IOSpec("negation_0", DataType.DtInt32) },
  98. new List<IOSpec> { new IOSpec("negation", DataType.DtInt32) },
  99. new List<EdgeSpec> { new EdgeSpec("negation_0", "neg:0"), new EdgeSpec("neg:y:0", "negation") },
  100. new List<EdgeSpec>());
  101. }
  102. [TestMethod]
  103. public void ZeroOps_Identity()
  104. {
  105. // Define
  106. var feed = Placeholder(func_graph_, s_);
  107. Define(-1,
  108. new Operation[0],
  109. new[] { feed },
  110. new[] { feed },
  111. new string[0]);
  112. // Use, run, and verify
  113. var func_feed = Placeholder(host_graph_, s_);
  114. var func_op = Use(new[] { func_feed });
  115. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) }, func_op, 3);
  116. VerifyFDef(new string[0],
  117. new List<IOSpec> { new IOSpec("feed_0", DataType.DtInt32) },
  118. new List<IOSpec> { new IOSpec("feed", DataType.DtInt32) },
  119. new List<EdgeSpec> { new EdgeSpec("feed_0", "feed") },
  120. new List<EdgeSpec>());
  121. }
  122. [TestMethod]
  123. public void ZeroOps_Permutation()
  124. {
  125. // Define
  126. var feed1 = Placeholder(func_graph_, s_, "feed1");
  127. var feed2 = Placeholder(func_graph_, s_, "feed2");
  128. Define(-1,
  129. null,
  130. new[] { feed1, feed2 },
  131. new[] { feed2, feed1 },
  132. null);
  133. // Use, run, and verify
  134. var two = ScalarConst(2, host_graph_, s_);
  135. var func_feed = Placeholder(host_graph_, s_);
  136. var func_op = Use(new[] { two, func_feed });
  137. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  138. new[] { new TF_Output(func_op, 0), new TF_Output(func_op, 1) },
  139. new[] { 3, 2 });
  140. VerifyFDef(new string[0],
  141. new List<IOSpec> { new IOSpec("feed1_0"), new IOSpec("feed2_0") },
  142. new List<IOSpec> { new IOSpec("feed2"), new IOSpec("feed1") },
  143. new List<EdgeSpec> { new EdgeSpec("feed1_0", "feed1"), new EdgeSpec("feed2_0", "feed2") },
  144. new List<EdgeSpec>());
  145. }
  146. [TestMethod]
  147. public void ZeroOps_Permutation_OutputNames()
  148. {
  149. // Define
  150. var feed1 = Placeholder(func_graph_, s_, "feed1");
  151. var feed2 = Placeholder(func_graph_, s_, "feed2");
  152. Define(-1,
  153. null,
  154. new[] { feed1, feed2 },
  155. new[] { feed2, feed1 },
  156. new[] { "first", "second" });
  157. // Use, run, and verify
  158. var two = ScalarConst(2, host_graph_, s_);
  159. var func_feed = Placeholder(host_graph_, s_);
  160. var func_op = Use(new[] { two, func_feed });
  161. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  162. new[] { new TF_Output(func_op, 0), new TF_Output(func_op, 1) },
  163. new[] { 3, 2 });
  164. VerifyFDef(new string[0],
  165. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2") },
  166. new List<IOSpec> { new IOSpec("first"), new IOSpec("second") },
  167. new List<EdgeSpec> { new EdgeSpec("feed1", "second"), new EdgeSpec("feed2", "first") },
  168. new List<EdgeSpec>());
  169. }
  170. [TestMethod]
  171. public void OneOp_TwoInputs_OneOutput()
  172. {
  173. // Define
  174. var feed1 = Placeholder(func_graph_, s_, "feed1");
  175. var feed2 = Placeholder(func_graph_, s_, "feed2");
  176. var add = Add(feed1, feed2, func_graph_, s_);
  177. Define(-1,
  178. null,
  179. new[] { feed1, feed2 },
  180. new[] { add },
  181. null);
  182. // Use, run, and verify
  183. var two = ScalarConst(2, host_graph_, s_);
  184. var func_feed = Placeholder(host_graph_, s_);
  185. var func_op = Use(new[] { two, func_feed });
  186. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  187. func_op,
  188. 2 + 3);
  189. VerifyFDef(new string[] { "add_0" },
  190. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2") },
  191. new List<IOSpec> { new IOSpec("add") },
  192. new List<EdgeSpec>
  193. {
  194. new EdgeSpec("feed1", "add_0:0"),
  195. new EdgeSpec("feed2", "add_0:1"),
  196. new EdgeSpec("add_0:sum:0", "add")
  197. },
  198. new List<EdgeSpec>());
  199. }
  200. [TestMethod]
  201. public void OneOp_TwoInputs_ZeroOutputs()
  202. {
  203. // Define
  204. var feed1 = Placeholder(func_graph_, s_, "feed1");
  205. var feed2 = Placeholder(func_graph_, s_, "feed2");
  206. var add = Add(feed1, feed2, func_graph_, s_);
  207. Define(-1,
  208. null,
  209. new[] { feed1, feed2 },
  210. new Operation[0],
  211. null);
  212. // Use, run, and verify
  213. var two = ScalarConst(2, host_graph_, s_);
  214. var func_feed = Placeholder(host_graph_, s_);
  215. var func_op = Use(new[] { two, func_feed });
  216. VerifyFDef(new string[] { "add" },
  217. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2") },
  218. new List<IOSpec>(),
  219. new List<EdgeSpec>
  220. {
  221. new EdgeSpec("feed1", "add:0"),
  222. new EdgeSpec("feed2", "add:1")
  223. },
  224. new List<EdgeSpec>());
  225. }
  226. [TestMethod]
  227. public void TwoOps_ThreeInputs_OneOutput()
  228. {
  229. // Define
  230. var feed1 = Placeholder(func_graph_, s_, "feed1");
  231. var feed2 = Placeholder(func_graph_, s_, "feed2");
  232. var feed3 = Placeholder(func_graph_, s_, "feed3");
  233. var add1 = Add(feed1, feed2, func_graph_, s_, "add1");
  234. var add2 = Add(add1, feed3, func_graph_, s_, "add2");
  235. Define(-1,
  236. null,
  237. new[] { feed1, feed2, feed3 },
  238. new[] { add2 },
  239. null);
  240. // Use, run, and verify
  241. var two = ScalarConst(2, host_graph_, s_, "two");
  242. var ten = ScalarConst(10, host_graph_, s_, "ten");
  243. var func_feed = Placeholder(host_graph_, s_);
  244. var func_op = Use(new[] { two, ten, func_feed });
  245. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  246. func_op,
  247. 2 + 10 + 3);
  248. VerifyFDef(new string[] { "add1", "add2_0" },
  249. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2"), new IOSpec("feed3") },
  250. new List<IOSpec> { new IOSpec("add2") },
  251. new List<EdgeSpec>
  252. {
  253. new EdgeSpec("feed1", "add1:0"),
  254. new EdgeSpec("feed2", "add1:1"),
  255. new EdgeSpec("add1:sum:0", "add2_0:0"),
  256. new EdgeSpec("feed3", "add2_0:1"),
  257. new EdgeSpec("add2_0:sum:0", "add2"),
  258. },
  259. new List<EdgeSpec>());
  260. }
  261. [TestMethod]
  262. public void OneOp_TwoInputs_TwoDuplicateOutputs()
  263. {
  264. // Define
  265. var feed1 = Placeholder(func_graph_, s_, "feed1");
  266. var feed2 = Placeholder(func_graph_, s_, "feed2");
  267. var add = Add(feed1, feed2, func_graph_, s_);
  268. Define(-1,
  269. null,
  270. new[] { feed1, feed2 },
  271. new[] { add, add },
  272. null);
  273. // Use, run, and verify
  274. var two = ScalarConst(2, host_graph_, s_);
  275. var func_feed = Placeholder(host_graph_, s_);
  276. var func_op = Use(new[] { two, func_feed });
  277. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  278. new[] { new TF_Output(func_op, 0), new TF_Output(func_op, 1) },
  279. new[] { 5, 5 });
  280. VerifyFDef(new string[] { "add_1" },
  281. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2") },
  282. new List<IOSpec> { new IOSpec("add"), new IOSpec("add_0") },
  283. new List<EdgeSpec>
  284. {
  285. new EdgeSpec("feed1", "add_1:0"),
  286. new EdgeSpec("feed2", "add_1:1"),
  287. new EdgeSpec("add_1:sum:0", "add"),
  288. new EdgeSpec("add_1:sum:0", "add_0")
  289. },
  290. new List<EdgeSpec>());
  291. }
  292. [TestMethod]
  293. public void TwoDuplicateOutputs_OutputNames()
  294. {
  295. // Define
  296. var feed1 = Placeholder(func_graph_, s_, "feed1");
  297. var feed2 = Placeholder(func_graph_, s_, "feed2");
  298. var add = Add(feed1, feed2, func_graph_, s_);
  299. Define(-1,
  300. null,
  301. new[] { feed1, feed2 },
  302. new[] { add, add },
  303. new[] { "out1", "out2" });
  304. // Use, run, and verify
  305. var two = ScalarConst(2, host_graph_, s_);
  306. var func_feed = Placeholder(host_graph_, s_);
  307. var func_op = Use(new[] { two, func_feed });
  308. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  309. new[] { new TF_Output(func_op, 0), new TF_Output(func_op, 1) },
  310. new[] { 5, 5 });
  311. VerifyFDef(new string[] { "add" },
  312. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2") },
  313. new List<IOSpec> { new IOSpec("out1"), new IOSpec("out2") },
  314. new List<EdgeSpec>
  315. {
  316. new EdgeSpec("feed1", "add:0"),
  317. new EdgeSpec("feed2", "add:1"),
  318. new EdgeSpec("add:sum:0", "out1"),
  319. new EdgeSpec("add:sum:0", "out2")
  320. },
  321. new List<EdgeSpec>());
  322. }
  323. [TestMethod]
  324. public void TwoOps_ThreeInputs_TwoOutputs()
  325. {
  326. // Define
  327. var feed1 = Placeholder(func_graph_, s_, "feed1");
  328. var feed2 = Placeholder(func_graph_, s_, "feed2");
  329. var feed3 = Placeholder(func_graph_, s_, "feed3");
  330. var add1 = Add(feed1, feed2, func_graph_, s_, "add1");
  331. var add2 = Add(add1, feed3, func_graph_, s_, "add2");
  332. Define(-1,
  333. null,
  334. new[] { feed1, feed2, feed3 },
  335. new[] { add1, add2 },
  336. null);
  337. // Use, run, and verify
  338. var two = ScalarConst(2, host_graph_, s_, "two");
  339. var ten = ScalarConst(10, host_graph_, s_, "ten");
  340. var func_feed = Placeholder(host_graph_, s_);
  341. var func_op = Use(new[] { two, ten, func_feed });
  342. Run(new[] { new KeyValuePair<Operation, Tensor>(func_feed, Int32Tensor(3)) },
  343. new[] { new TF_Output(func_op, 0), new TF_Output(func_op, 1) },
  344. new[] { 12, 15 });
  345. VerifyFDef(new string[] { "add1_0", "add2_0" },
  346. new List<IOSpec> { new IOSpec("feed1"), new IOSpec("feed2"), new IOSpec("feed3") },
  347. new List<IOSpec> { new IOSpec("add1"), new IOSpec("add2") },
  348. new List<EdgeSpec>
  349. {
  350. new EdgeSpec("feed1", "add1_0:0"),
  351. new EdgeSpec("feed2", "add1_0:1"),
  352. new EdgeSpec("add1_0:sum:0", "add2_0:0"),
  353. new EdgeSpec("feed3", "add2_0:1"),
  354. new EdgeSpec("add1_0:sum:0", "add1"),
  355. new EdgeSpec("add2_0:sum:0", "add2")
  356. },
  357. new List<EdgeSpec>());
  358. }
  359. void Define(int num_opers, Operation[] opers,
  360. Operation[] inputs, Operation[] outputs,
  361. string[] output_names, bool expect_failure = false)
  362. => DefineT(num_opers, opers,
  363. inputs.Select(x => new TF_Output(x, 0)).ToArray(),
  364. outputs.Select(x => new TF_Output(x, 0)).ToArray(),
  365. output_names, expect_failure);
  366. void DefineT(int num_opers, Operation[] opers,
  367. TF_Output[] inputs, TF_Output[] outputs,
  368. string[] output_names, bool expect_failure = false)
  369. {
  370. func_ = c_api.TF_GraphToFunction(func_graph_, func_name_, false,
  371. num_opers, num_opers == -1 ? null : opers.Select(x => (IntPtr)x).ToArray(),
  372. inputs.Length, inputs.ToArray(),
  373. outputs.Length, outputs.ToArray(),
  374. output_names == null || output_names.Length == 0 ? null : output_names,
  375. IntPtr.Zero, null, s_.Handle);
  376. if (expect_failure)
  377. {
  378. ASSERT_EQ(IntPtr.Zero, func_);
  379. return;
  380. }
  381. ASSERT_EQ(TF_OK, s_.Code, s_.Message);
  382. ASSERT_NE(func_, IntPtr.Zero);
  383. ASSERT_EQ(func_name_, c_api.StringPiece(c_api.TF_FunctionName(func_)));
  384. c_api.TF_GraphCopyFunction(host_graph_, func_, IntPtr.Zero, s_.Handle);
  385. ASSERT_EQ(TF_OK, s_.Code, s_.Message);
  386. }
  387. Operation Use(Operation[] inputs)
  388. => UseT(inputs.Select(x => new TF_Output(x, 0)).ToArray());
  389. Operation UseT(TF_Output[] inputs)
  390. => UseHelper(inputs);
  391. Operation UseHelper(TF_Output[] inputs)
  392. {
  393. var desc = TF_NewOperation(host_graph_, func_name_, func_node_name_);
  394. foreach (var input in inputs)
  395. TF_AddInput(desc, input);
  396. c_api.TF_SetDevice(desc, "/cpu:0");
  397. var op = TF_FinishOperation(desc, s_);
  398. ASSERT_EQ(TF_OK, s_.Code, s_.Message);
  399. ASSERT_NE(op, IntPtr.Zero);
  400. return op;
  401. }
  402. void Run(KeyValuePair<Operation, Tensor>[] inputs, Operation output, int expected_result)
  403. => Run(inputs, new[] { new TF_Output(output, 0) }, new[] { expected_result });
  404. unsafe void Run(KeyValuePair<Operation, Tensor>[] inputs, TF_Output[] outputs, int[] expected_results)
  405. {
  406. var csession = new CSession(host_graph_, s_);
  407. ASSERT_EQ(TF_OK, s_.Code, s_.Message);
  408. csession.SetInputs(inputs);
  409. csession.SetOutputs(outputs);
  410. csession.Run(s_);
  411. ASSERT_EQ(TF_OK, s_.Code, s_.Message);
  412. for (int i = 0; i < expected_results.Length; ++i)
  413. {
  414. var output = csession.output_tensor(i);
  415. ASSERT_TRUE(output != IntPtr.Zero);
  416. EXPECT_EQ(TF_DataType.TF_INT32, c_api.TF_TensorType(output));
  417. EXPECT_EQ(0, c_api.TF_NumDims(output));
  418. ASSERT_EQ(sizeof(int), (int)c_api.TF_TensorByteSize(output));
  419. var output_contents = c_api.TF_TensorData(output);
  420. EXPECT_EQ(expected_results[i], *(int*)output_contents.ToPointer());
  421. }
  422. }
  423. void VerifyFDef(string[] nodes, List<IOSpec> inputs, List<IOSpec> outputs,
  424. List<EdgeSpec> e_edges, List<EdgeSpec> c_edges,
  425. bool is_exact_edges = true)
  426. {
  427. var fdef = GetFunctionDef(func_);
  428. EXPECT_NE(fdef, IntPtr.Zero);
  429. VerifyFDefNodes(fdef, nodes);
  430. VerifyFDefInputs(fdef, inputs);
  431. VerifyFDefOutputs(fdef, outputs);
  432. VerifyFDefEdges(fdef, e_edges, c_edges, is_exact_edges);
  433. }
  434. void VerifyFDefNodes(FunctionDef fdef, string[] nodes)
  435. {
  436. ASSERT_EQ(nodes.Length, fdef.NodeDef.Count);
  437. foreach (var node in fdef.NodeDef)
  438. {
  439. ASSERT_TRUE(nodes.Contains(node.Name), $"Got unexpected node: {node.Name} in fdef: {fdef}");
  440. }
  441. }
  442. void VerifyFDefInputs(FunctionDef fdef, List<IOSpec> inputs)
  443. {
  444. var signature = fdef.Signature;
  445. ASSERT_EQ(inputs.Count, signature.InputArg.Count);
  446. for (int i = 0; i < inputs.Count; ++i)
  447. {
  448. var arg = signature.InputArg[i];
  449. var input = inputs[i];
  450. if (input.Value != DataType.DtInvalid)
  451. ASSERT_EQ(arg.Type, input.Value, $"");
  452. ASSERT_EQ(arg.Name, input.Key, $"Got unexpected name for input {i}. fdef: {fdef}");
  453. }
  454. }
  455. void VerifyFDefOutputs(FunctionDef fdef, List<IOSpec> outputs)
  456. {
  457. var signature = fdef.Signature;
  458. ASSERT_EQ(outputs.Count, signature.OutputArg.Count);
  459. for (int i = 0; i < outputs.Count; ++i)
  460. {
  461. var arg = signature.OutputArg[i];
  462. var output = outputs[i];
  463. if (output.Value != DataType.DtInvalid)
  464. ASSERT_EQ(arg.Type, output.Value, $"");
  465. ASSERT_EQ(arg.Name, output.Key, $"Got unexpected name for input {i}. fdef: {fdef}");
  466. }
  467. }
  468. void VerifyFDefEdges(FunctionDef fdef, List<EdgeSpec> e_edges, List<EdgeSpec> c_edges, bool is_exact_edges = true)
  469. {
  470. // Build a set of edges from fdef
  471. var a_edges = new List<EdgeSpec>(); // actual edges
  472. // Get edges from inputs to body nodes and between body nodes
  473. foreach (var node in fdef.NodeDef)
  474. {
  475. for (int i = 0; i < node.Input.Count; ++i)
  476. {
  477. var input = node.Input[i];
  478. a_edges.Add(new EdgeSpec(input, $"{node.Name}:{i}"));
  479. }
  480. }
  481. // Get edges from body nodes to outputs and from inputs to outputs
  482. foreach (var arg in fdef.Signature.OutputArg)
  483. {
  484. var iter = fdef.Ret.FirstOrDefault(x => x.Key == arg.Name);
  485. if (iter.Key != null)
  486. {
  487. a_edges.Add(new EdgeSpec(iter.Value, arg.Name));
  488. }
  489. else
  490. {
  491. a_edges.Add(new EdgeSpec(arg.Name, arg.Name));
  492. }
  493. }
  494. // Verify edges
  495. foreach (var edge in e_edges)
  496. {
  497. ASSERT_TRUE(a_edges.Contains(edge));
  498. }
  499. foreach (var edge in c_edges)
  500. {
  501. ASSERT_TRUE(a_edges.Contains(edge));
  502. }
  503. // If caller specified all edges, check that we have seen all
  504. if (is_exact_edges)
  505. {
  506. ASSERT_EQ(e_edges.Count + c_edges.Count, a_edges.Count,
  507. $"Expected edges: {e_edges}, Expected Control edges: {c_edges}, Actual edges: {a_edges}");
  508. }
  509. }
  510. public void Dispose()
  511. {
  512. }
  513. public struct IOSpec
  514. {
  515. KeyValuePair<string, DataType> pair;
  516. public string Key => pair.Key;
  517. public DataType Value => pair.Value;
  518. public IOSpec(string key, DataType value = DataType.DtInvalid)
  519. {
  520. pair = new KeyValuePair<string, DataType>(key, value);
  521. }
  522. }
  523. public struct EdgeSpec
  524. {
  525. KeyValuePair<string, string> pair;
  526. public string Key => pair.Key;
  527. public string Value => pair.Value;
  528. public EdgeSpec(string key, string value)
  529. {
  530. pair = new KeyValuePair<string, string>(key, value);
  531. }
  532. }
  533. }
  534. }