1681c28a3SPeter Maydell #include "qemu/osdep.h" 26b673957SMarkus Armbruster #include "qapi/qmp/qdict.h" 315280c36SMarkus Armbruster #include "qapi/qmp/qnum.h" 46b673957SMarkus Armbruster #include "qapi/qmp/qstring.h" 5e688df6bSMarkus Armbruster #include "qapi/error.h" 6b3db211fSDaniel P. Berrange #include "qapi/qobject-input-visitor.h" 7d98150f0SLaszlo Ersek #include "tests/test-qapi-types.h" 8d98150f0SLaszlo Ersek #include "tests/test-qapi-visit.h" 9eb815e24SMarkus Armbruster #include "test-qapi-commands.h" 1000ca24ffSMarkus Armbruster #include "test-qapi-init-commands.h" 1169ed8366SMichael Roth 121527badbSMarkus Armbruster static QmpCommandList qmp_commands; 131527badbSMarkus Armbruster 141f7b9f31SMarc-André Lureau #if defined(TEST_IF_STRUCT) && defined(TEST_IF_CMD) 15967c8851SMarc-André Lureau UserDefThree *qmp_TestIfCmd(TestIfStruct *foo, Error **errp) 16967c8851SMarc-André Lureau { 17967c8851SMarc-André Lureau return NULL; 18967c8851SMarc-André Lureau } 191f7b9f31SMarc-André Lureau #endif 20967c8851SMarc-André Lureau 21967c8851SMarc-André Lureau UserDefThree *qmp_TestCmdReturnDefThree(Error **errp) 22967c8851SMarc-André Lureau { 23967c8851SMarc-André Lureau return NULL; 24967c8851SMarc-André Lureau } 25967c8851SMarc-André Lureau 2669ed8366SMichael Roth void qmp_user_def_cmd(Error **errp) 2769ed8366SMichael Roth { 2869ed8366SMichael Roth } 2969ed8366SMichael Roth 307b13f2c2SIgor Mammedov void qmp_test_flags_command(Error **errp) 311a1b11dcSPeter Xu { 321a1b11dcSPeter Xu } 331a1b11dcSPeter Xu 34ae6bf766SMarc-André Lureau void qmp_cmd_success_response(Error **errp) 35ae6bf766SMarc-André Lureau { 36ae6bf766SMarc-André Lureau } 37ae6bf766SMarc-André Lureau 38972a1101SEric Blake Empty2 *qmp_user_def_cmd0(Error **errp) 39972a1101SEric Blake { 40972a1101SEric Blake return g_new0(Empty2, 1); 41972a1101SEric Blake } 42972a1101SEric Blake 4369ed8366SMichael Roth void qmp_user_def_cmd1(UserDefOne * ud1, Error **errp) 4469ed8366SMichael Roth { 4569ed8366SMichael Roth } 4669ed8366SMichael Roth 478aa3a33eSKevin Wolf void qmp_test_features(FeatureStruct0 *fs0, FeatureStruct1 *fs1, 488aa3a33eSKevin Wolf FeatureStruct2 *fs2, FeatureStruct3 *fs3, 498aa3a33eSKevin Wolf FeatureStruct4 *fs4, CondFeatureStruct1 *cfs1, 508aa3a33eSKevin Wolf CondFeatureStruct2 *cfs2, CondFeatureStruct3 *cfs3, 518aa3a33eSKevin Wolf Error **errp) 528aa3a33eSKevin Wolf { 538aa3a33eSKevin Wolf } 548aa3a33eSKevin Wolf 552e2e0df2SPeter Krempa void qmp_test_command_features0(Error **errp) 562e2e0df2SPeter Krempa { 572e2e0df2SPeter Krempa } 582e2e0df2SPeter Krempa 592e2e0df2SPeter Krempa void qmp_test_command_features1(Error **errp) 602e2e0df2SPeter Krempa { 612e2e0df2SPeter Krempa } 622e2e0df2SPeter Krempa 632e2e0df2SPeter Krempa void qmp_test_command_features3(Error **errp) 642e2e0df2SPeter Krempa { 652e2e0df2SPeter Krempa } 662e2e0df2SPeter Krempa 672e2e0df2SPeter Krempa void qmp_test_command_cond_features1(Error **errp) 682e2e0df2SPeter Krempa { 692e2e0df2SPeter Krempa } 702e2e0df2SPeter Krempa 712e2e0df2SPeter Krempa void qmp_test_command_cond_features2(Error **errp) 722e2e0df2SPeter Krempa { 732e2e0df2SPeter Krempa } 742e2e0df2SPeter Krempa 752e2e0df2SPeter Krempa void qmp_test_command_cond_features3(Error **errp) 762e2e0df2SPeter Krempa { 772e2e0df2SPeter Krempa } 782e2e0df2SPeter Krempa 79ab22ad96SMarkus Armbruster UserDefTwo *qmp_user_def_cmd2(UserDefOne *ud1a, 80ab22ad96SMarkus Armbruster bool has_udb1, UserDefOne *ud1b, 81ab22ad96SMarkus Armbruster Error **errp) 8269ed8366SMichael Roth { 8369ed8366SMichael Roth UserDefTwo *ret; 847267c094SAnthony Liguori UserDefOne *ud1c = g_malloc0(sizeof(UserDefOne)); 857267c094SAnthony Liguori UserDefOne *ud1d = g_malloc0(sizeof(UserDefOne)); 8669ed8366SMichael Roth 8769ed8366SMichael Roth ud1c->string = strdup(ud1a->string); 88ddf21908SEric Blake ud1c->integer = ud1a->integer; 89ab22ad96SMarkus Armbruster ud1d->string = strdup(has_udb1 ? ud1b->string : "blah0"); 90ddf21908SEric Blake ud1d->integer = has_udb1 ? ud1b->integer : 0; 9169ed8366SMichael Roth 92b6fcf32dSEric Blake ret = g_new0(UserDefTwo, 1); 93b6fcf32dSEric Blake ret->string0 = strdup("blah1"); 946446a592SEric Blake ret->dict1 = g_new0(UserDefTwoDict, 1); 956446a592SEric Blake ret->dict1->string1 = strdup("blah2"); 966446a592SEric Blake ret->dict1->dict2 = g_new0(UserDefTwoDictDict, 1); 976446a592SEric Blake ret->dict1->dict2->userdef = ud1c; 986446a592SEric Blake ret->dict1->dict2->string = strdup("blah3"); 996446a592SEric Blake ret->dict1->dict3 = g_new0(UserDefTwoDictDict, 1); 1006446a592SEric Blake ret->dict1->has_dict3 = true; 1016446a592SEric Blake ret->dict1->dict3->userdef = ud1d; 1026446a592SEric Blake ret->dict1->dict3->string = strdup("blah4"); 10369ed8366SMichael Roth 10469ed8366SMichael Roth return ret; 10569ed8366SMichael Roth } 10669ed8366SMichael Roth 107cae95eaeSEric Blake int64_t qmp_guest_get_time(int64_t a, bool has_b, int64_t b, Error **errp) 108c2216a8aSMarkus Armbruster { 109c2216a8aSMarkus Armbruster return a + (has_b ? b : 0); 110c2216a8aSMarkus Armbruster } 111c2216a8aSMarkus Armbruster 11228770e05SMarkus Armbruster QObject *qmp_guest_sync(QObject *arg, Error **errp) 11328770e05SMarkus Armbruster { 11428770e05SMarkus Armbruster return arg; 11528770e05SMarkus Armbruster } 11628770e05SMarkus Armbruster 117c818408eSEric Blake void qmp_boxed_struct(UserDefZero *arg, Error **errp) 118c818408eSEric Blake { 119c818408eSEric Blake } 120c818408eSEric Blake 121b359f4b2SMarkus Armbruster void qmp_boxed_union(UserDefListUnion *arg, Error **errp) 122c818408eSEric Blake { 123c818408eSEric Blake } 124c818408eSEric Blake 125675b214bSMarkus Armbruster void qmp_boxed_empty(Empty1 *arg, Error **errp) 126675b214bSMarkus Armbruster { 127675b214bSMarkus Armbruster } 128675b214bSMarkus Armbruster 129e3c4c3d7SEric Blake __org_qemu_x_Union1 *qmp___org_qemu_x_command(__org_qemu_x_EnumList *a, 130e3c4c3d7SEric Blake __org_qemu_x_StructList *b, 131e3c4c3d7SEric Blake __org_qemu_x_Union2 *c, 132e3c4c3d7SEric Blake __org_qemu_x_Alt *d, 133e3c4c3d7SEric Blake Error **errp) 134e3c4c3d7SEric Blake { 135e3c4c3d7SEric Blake __org_qemu_x_Union1 *ret = g_new0(__org_qemu_x_Union1, 1); 136e3c4c3d7SEric Blake 137c363acefSEric Blake ret->type = ORG_QEMU_X_UNION1_KIND___ORG_QEMU_X_BRANCH; 13832bafa8fSEric Blake ret->u.__org_qemu_x_branch.data = strdup("blah1"); 139e3c4c3d7SEric Blake 140c43567c1SEric Blake /* Also test that 'wchar-t' was munged to 'q_wchar_t' */ 141c43567c1SEric Blake if (b && b->value && !b->value->has_q_wchar_t) { 142c43567c1SEric Blake b->value->q_wchar_t = 1; 143c43567c1SEric Blake } 144e3c4c3d7SEric Blake return ret; 145e3c4c3d7SEric Blake } 146e3c4c3d7SEric Blake 147e3c4c3d7SEric Blake 148*3306459aSMarkus Armbruster static QObject *do_qmp_dispatch(QDict *req, bool allow_oob) 149*3306459aSMarkus Armbruster { 150*3306459aSMarkus Armbruster QDict *resp; 151*3306459aSMarkus Armbruster QObject *ret; 152*3306459aSMarkus Armbruster 153*3306459aSMarkus Armbruster resp = qmp_dispatch(&qmp_commands, QOBJECT(req), allow_oob); 154*3306459aSMarkus Armbruster g_assert(resp && !qdict_haskey(resp, "error")); 155*3306459aSMarkus Armbruster ret = qdict_get(resp, "return"); 156*3306459aSMarkus Armbruster g_assert(ret); 157*3306459aSMarkus Armbruster 158*3306459aSMarkus Armbruster qobject_ref(ret); 159*3306459aSMarkus Armbruster qobject_unref(resp); 160*3306459aSMarkus Armbruster return ret; 161*3306459aSMarkus Armbruster } 162*3306459aSMarkus Armbruster 163*3306459aSMarkus Armbruster static void do_qmp_dispatch_error(QDict *req, bool allow_oob, ErrorClass cls) 164*3306459aSMarkus Armbruster { 165*3306459aSMarkus Armbruster QDict *resp; 166*3306459aSMarkus Armbruster 167*3306459aSMarkus Armbruster resp = qmp_dispatch(&qmp_commands, QOBJECT(req), allow_oob); 168*3306459aSMarkus Armbruster g_assert(resp && qdict_haskey(resp, "error")); 169*3306459aSMarkus Armbruster 170*3306459aSMarkus Armbruster qobject_unref(resp); 171*3306459aSMarkus Armbruster } 172*3306459aSMarkus Armbruster 17369ed8366SMichael Roth /* test commands with no input and no return value */ 17469ed8366SMichael Roth static void test_dispatch_cmd(void) 17569ed8366SMichael Roth { 17669ed8366SMichael Roth QDict *req = qdict_new(); 177*3306459aSMarkus Armbruster QObject *ret; 17869ed8366SMichael Roth 17946f5ac20SEric Blake qdict_put_str(req, "execute", "user_def_cmd"); 18069ed8366SMichael Roth 181*3306459aSMarkus Armbruster ret = do_qmp_dispatch(req, false); 18269ed8366SMichael Roth 183*3306459aSMarkus Armbruster qobject_unref(ret); 184cb3e7f08SMarc-André Lureau qobject_unref(req); 18569ed8366SMichael Roth } 18669ed8366SMichael Roth 187c55f070bSPeter Xu static void test_dispatch_cmd_oob(void) 188c55f070bSPeter Xu { 189c55f070bSPeter Xu QDict *req = qdict_new(); 190*3306459aSMarkus Armbruster QObject *ret; 191c55f070bSPeter Xu 192c55f070bSPeter Xu qdict_put_str(req, "exec-oob", "test-flags-command"); 193c55f070bSPeter Xu 194*3306459aSMarkus Armbruster ret = do_qmp_dispatch(req, true); 195c55f070bSPeter Xu 196*3306459aSMarkus Armbruster qobject_unref(ret); 197c55f070bSPeter Xu qobject_unref(req); 198c55f070bSPeter Xu } 199c55f070bSPeter Xu 20069ed8366SMichael Roth /* test commands that return an error due to invalid parameters */ 20151009170SPeter Maydell static void test_dispatch_cmd_failure(void) 20269ed8366SMichael Roth { 20369ed8366SMichael Roth QDict *req = qdict_new(); 204a0067da1SMarc-André Lureau QDict *args = qdict_new(); 20569ed8366SMichael Roth 20646f5ac20SEric Blake qdict_put_str(req, "execute", "user_def_cmd2"); 20769ed8366SMichael Roth 208*3306459aSMarkus Armbruster do_qmp_dispatch_error(req, false, ERROR_CLASS_GENERIC_ERROR); 20969ed8366SMichael Roth 210cb3e7f08SMarc-André Lureau qobject_unref(req); 211a0067da1SMarc-André Lureau 212a0067da1SMarc-André Lureau /* check that with extra arguments it throws an error */ 213a0067da1SMarc-André Lureau req = qdict_new(); 21446f5ac20SEric Blake qdict_put_int(args, "a", 66); 215a0067da1SMarc-André Lureau qdict_put(req, "arguments", args); 216a0067da1SMarc-André Lureau 21746f5ac20SEric Blake qdict_put_str(req, "execute", "user_def_cmd"); 218a0067da1SMarc-André Lureau 219*3306459aSMarkus Armbruster do_qmp_dispatch_error(req, false, ERROR_CLASS_GENERIC_ERROR); 220a0067da1SMarc-André Lureau 221cb3e7f08SMarc-André Lureau qobject_unref(req); 22269ed8366SMichael Roth } 22369ed8366SMichael Roth 224ae6bf766SMarc-André Lureau static void test_dispatch_cmd_success_response(void) 225ae6bf766SMarc-André Lureau { 226ae6bf766SMarc-André Lureau QDict *req = qdict_new(); 227ae6bf766SMarc-André Lureau QDict *resp; 228ae6bf766SMarc-André Lureau 229ae6bf766SMarc-André Lureau qdict_put_str(req, "execute", "cmd-success-response"); 230ae6bf766SMarc-André Lureau resp = qmp_dispatch(&qmp_commands, QOBJECT(req), false); 231ae6bf766SMarc-André Lureau g_assert_null(resp); 232ae6bf766SMarc-André Lureau qobject_unref(req); 233ae6bf766SMarc-André Lureau } 234ae6bf766SMarc-André Lureau 23569ed8366SMichael Roth /* test commands that involve both input parameters and return values */ 23669ed8366SMichael Roth static void test_dispatch_cmd_io(void) 23769ed8366SMichael Roth { 23869ed8366SMichael Roth QDict *req = qdict_new(); 23969ed8366SMichael Roth QDict *args = qdict_new(); 240c2216a8aSMarkus Armbruster QDict *args3 = qdict_new(); 24169ed8366SMichael Roth QDict *ud1a = qdict_new(); 24269ed8366SMichael Roth QDict *ud1b = qdict_new(); 243357765feSMarkus Armbruster QDict *ret, *ret_dict, *ret_dict_dict, *ret_dict_dict_userdef; 244357765feSMarkus Armbruster QDict *ret_dict_dict2, *ret_dict_dict2_userdef; 24501b2ffceSMarc-André Lureau QNum *ret3; 24601b2ffceSMarc-André Lureau int64_t val; 24769ed8366SMichael Roth 24846f5ac20SEric Blake qdict_put_int(ud1a, "integer", 42); 24946f5ac20SEric Blake qdict_put_str(ud1a, "string", "hello"); 25046f5ac20SEric Blake qdict_put_int(ud1b, "integer", 422); 25146f5ac20SEric Blake qdict_put_str(ud1b, "string", "hello2"); 252de6e7951SEric Blake qdict_put(args, "ud1a", ud1a); 253de6e7951SEric Blake qdict_put(args, "ud1b", ud1b); 254de6e7951SEric Blake qdict_put(req, "arguments", args); 25546f5ac20SEric Blake qdict_put_str(req, "execute", "user_def_cmd2"); 25669ed8366SMichael Roth 257*3306459aSMarkus Armbruster ret = qobject_to(QDict, do_qmp_dispatch(req, false)); 25869ed8366SMichael Roth 259b6fcf32dSEric Blake assert(!strcmp(qdict_get_str(ret, "string0"), "blah1")); 260b6fcf32dSEric Blake ret_dict = qdict_get_qdict(ret, "dict1"); 261b6fcf32dSEric Blake assert(!strcmp(qdict_get_str(ret_dict, "string1"), "blah2")); 262b6fcf32dSEric Blake ret_dict_dict = qdict_get_qdict(ret_dict, "dict2"); 263357765feSMarkus Armbruster ret_dict_dict_userdef = qdict_get_qdict(ret_dict_dict, "userdef"); 264357765feSMarkus Armbruster assert(qdict_get_int(ret_dict_dict_userdef, "integer") == 42); 265357765feSMarkus Armbruster assert(!strcmp(qdict_get_str(ret_dict_dict_userdef, "string"), "hello")); 266357765feSMarkus Armbruster assert(!strcmp(qdict_get_str(ret_dict_dict, "string"), "blah3")); 267b6fcf32dSEric Blake ret_dict_dict2 = qdict_get_qdict(ret_dict, "dict3"); 268357765feSMarkus Armbruster ret_dict_dict2_userdef = qdict_get_qdict(ret_dict_dict2, "userdef"); 269357765feSMarkus Armbruster assert(qdict_get_int(ret_dict_dict2_userdef, "integer") == 422); 270357765feSMarkus Armbruster assert(!strcmp(qdict_get_str(ret_dict_dict2_userdef, "string"), "hello2")); 271357765feSMarkus Armbruster assert(!strcmp(qdict_get_str(ret_dict_dict2, "string"), "blah4")); 272cb3e7f08SMarc-André Lureau qobject_unref(ret); 273c2216a8aSMarkus Armbruster 27446f5ac20SEric Blake qdict_put_int(args3, "a", 66); 275c2216a8aSMarkus Armbruster qdict_put(req, "arguments", args3); 27646f5ac20SEric Blake qdict_put_str(req, "execute", "guest-get-time"); 277c2216a8aSMarkus Armbruster 278*3306459aSMarkus Armbruster ret3 = qobject_to(QNum, do_qmp_dispatch(req, false)); 27901b2ffceSMarc-André Lureau g_assert(qnum_get_try_int(ret3, &val)); 28001b2ffceSMarc-André Lureau g_assert_cmpint(val, ==, 66); 281cb3e7f08SMarc-André Lureau qobject_unref(ret3); 282c2216a8aSMarkus Armbruster 283cb3e7f08SMarc-André Lureau qobject_unref(req); 28469ed8366SMichael Roth } 28569ed8366SMichael Roth 2865cd5f0d0SMichael Roth /* test generated dealloc functions for generated types */ 2875cd5f0d0SMichael Roth static void test_dealloc_types(void) 2885cd5f0d0SMichael Roth { 2895cd5f0d0SMichael Roth UserDefOne *ud1test, *ud1a, *ud1b; 2905cd5f0d0SMichael Roth UserDefOneList *ud1list; 2915cd5f0d0SMichael Roth 2925cd5f0d0SMichael Roth ud1test = g_malloc0(sizeof(UserDefOne)); 293ddf21908SEric Blake ud1test->integer = 42; 2945cd5f0d0SMichael Roth ud1test->string = g_strdup("hi there 42"); 2955cd5f0d0SMichael Roth 2965cd5f0d0SMichael Roth qapi_free_UserDefOne(ud1test); 2975cd5f0d0SMichael Roth 2985cd5f0d0SMichael Roth ud1a = g_malloc0(sizeof(UserDefOne)); 299ddf21908SEric Blake ud1a->integer = 43; 3005cd5f0d0SMichael Roth ud1a->string = g_strdup("hi there 43"); 3015cd5f0d0SMichael Roth 3025cd5f0d0SMichael Roth ud1b = g_malloc0(sizeof(UserDefOne)); 303ddf21908SEric Blake ud1b->integer = 44; 3045cd5f0d0SMichael Roth ud1b->string = g_strdup("hi there 44"); 3055cd5f0d0SMichael Roth 3065cd5f0d0SMichael Roth ud1list = g_malloc0(sizeof(UserDefOneList)); 3075cd5f0d0SMichael Roth ud1list->value = ud1a; 3085cd5f0d0SMichael Roth ud1list->next = g_malloc0(sizeof(UserDefOneList)); 3095cd5f0d0SMichael Roth ud1list->next->value = ud1b; 3105cd5f0d0SMichael Roth 3115cd5f0d0SMichael Roth qapi_free_UserDefOneList(ud1list); 3125cd5f0d0SMichael Roth } 3135cd5f0d0SMichael Roth 314d98150f0SLaszlo Ersek /* test generated deallocation on an object whose construction was prematurely 315d98150f0SLaszlo Ersek * terminated due to an error */ 316d98150f0SLaszlo Ersek static void test_dealloc_partial(void) 317d98150f0SLaszlo Ersek { 318d98150f0SLaszlo Ersek static const char text[] = "don't leak me"; 319d98150f0SLaszlo Ersek 320d98150f0SLaszlo Ersek UserDefTwo *ud2 = NULL; 321d98150f0SLaszlo Ersek Error *err = NULL; 322d98150f0SLaszlo Ersek 323d98150f0SLaszlo Ersek /* create partial object */ 324d98150f0SLaszlo Ersek { 325d98150f0SLaszlo Ersek QDict *ud2_dict; 326b70ce101SEric Blake Visitor *v; 327d98150f0SLaszlo Ersek 328d98150f0SLaszlo Ersek ud2_dict = qdict_new(); 32946f5ac20SEric Blake qdict_put_str(ud2_dict, "string0", text); 330d98150f0SLaszlo Ersek 331048abb7bSMarkus Armbruster v = qobject_input_visitor_new(QOBJECT(ud2_dict)); 332b70ce101SEric Blake visit_type_UserDefTwo(v, NULL, &ud2, &err); 333b70ce101SEric Blake visit_free(v); 334cb3e7f08SMarc-André Lureau qobject_unref(ud2_dict); 335d98150f0SLaszlo Ersek } 336d98150f0SLaszlo Ersek 33768ab47e4SEric Blake /* verify that visit_type_XXX() cleans up properly on error */ 338a12a5a1aSEric Blake error_free_or_abort(&err); 33968ab47e4SEric Blake assert(!ud2); 34068ab47e4SEric Blake 34168ab47e4SEric Blake /* Manually create a partial object, leaving ud2->dict1 at NULL */ 34268ab47e4SEric Blake ud2 = g_new0(UserDefTwo, 1); 34368ab47e4SEric Blake ud2->string0 = g_strdup(text); 344d98150f0SLaszlo Ersek 345d98150f0SLaszlo Ersek /* tear down partial object */ 346d98150f0SLaszlo Ersek qapi_free_UserDefTwo(ud2); 347d98150f0SLaszlo Ersek } 348d98150f0SLaszlo Ersek 349d98150f0SLaszlo Ersek 35069ed8366SMichael Roth int main(int argc, char **argv) 35169ed8366SMichael Roth { 35269ed8366SMichael Roth g_test_init(&argc, &argv, NULL); 35369ed8366SMichael Roth 354214e4a5bSMarc-André Lureau g_test_add_func("/qmp/dispatch_cmd", test_dispatch_cmd); 355c55f070bSPeter Xu g_test_add_func("/qmp/dispatch_cmd_oob", test_dispatch_cmd_oob); 356214e4a5bSMarc-André Lureau g_test_add_func("/qmp/dispatch_cmd_failure", test_dispatch_cmd_failure); 357214e4a5bSMarc-André Lureau g_test_add_func("/qmp/dispatch_cmd_io", test_dispatch_cmd_io); 358ae6bf766SMarc-André Lureau g_test_add_func("/qmp/dispatch_cmd_success_response", 359ae6bf766SMarc-André Lureau test_dispatch_cmd_success_response); 360214e4a5bSMarc-André Lureau g_test_add_func("/qmp/dealloc_types", test_dealloc_types); 361214e4a5bSMarc-André Lureau g_test_add_func("/qmp/dealloc_partial", test_dealloc_partial); 36269ed8366SMichael Roth 3631527badbSMarkus Armbruster test_qmp_init_marshal(&qmp_commands); 36469ed8366SMichael Roth g_test_run(); 36569ed8366SMichael Roth 36669ed8366SMichael Roth return 0; 36769ed8366SMichael Roth } 368