Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1788)

Side by Side Diff: test/cctest/test-api.cc

Issue 376223002: Refactor ScriptData class for cached compile data. (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Created 6 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 14796 matching lines...) Expand 10 before | Expand all | Expand 10 after
14807 v8::Isolate* isolate = env->GetIsolate(); 14807 v8::Isolate* isolate = env->GetIsolate();
14808 HandleScope handle_scope(isolate); 14808 HandleScope handle_scope(isolate);
14809 14809
14810 i::FLAG_min_preparse_length = 0; 14810 i::FLAG_min_preparse_length = 0;
14811 const char* script = "function foo(a) { return a+1; }"; 14811 const char* script = "function foo(a) { return a+1; }";
14812 v8::ScriptCompiler::Source source(v8_str(script)); 14812 v8::ScriptCompiler::Source source(v8_str(script));
14813 v8::ScriptCompiler::Compile(isolate, &source, 14813 v8::ScriptCompiler::Compile(isolate, &source,
14814 v8::ScriptCompiler::kProduceDataToCache); 14814 v8::ScriptCompiler::kProduceDataToCache);
14815 // Serialize. 14815 // Serialize.
14816 const v8::ScriptCompiler::CachedData* cd = source.GetCachedData(); 14816 const v8::ScriptCompiler::CachedData* cd = source.GetCachedData();
14817 char* serialized_data = i::NewArray<char>(cd->length); 14817 i::byte* serialized_data = i::NewArray<i::byte>(cd->length);
14818 i::MemCopy(serialized_data, cd->data, cd->length); 14818 i::MemCopy(serialized_data, cd->data, cd->length);
14819 14819
14820 // Deserialize. 14820 // Deserialize.
14821 i::ScriptData* deserialized = i::ScriptData::New(serialized_data, cd->length); 14821 i::ScriptData* deserialized = new i::ScriptData(serialized_data, cd->length);
14822 14822
14823 // Verify that the original is the same as the deserialized. 14823 // Verify that the original is the same as the deserialized.
14824 CHECK_EQ(cd->length, deserialized->Length()); 14824 CHECK_EQ(cd->length, deserialized->length());
14825 CHECK_EQ(0, memcmp(cd->data, deserialized->Data(), cd->length)); 14825 CHECK_EQ(0, memcmp(cd->data, deserialized->data(), cd->length));
14826 14826
14827 delete deserialized; 14827 delete deserialized;
14828 i::DeleteArray(serialized_data); 14828 i::DeleteArray(serialized_data);
14829 } 14829 }
14830 14830
14831 14831
14832 // Attempts to deserialize bad data. 14832 // Attempts to deserialize bad data.
14833 TEST(PreCompileDeserializationError) { 14833 TEST(ExpectFailPreCompileDeserializationError) {
marja 2014/07/09 14:42:30 I'd just remove this test, it makes no sense to ke
Yang 2014/07/10 08:28:46 Done.
14834 v8::V8::Initialize(); 14834 v8::V8::Initialize();
14835 const char* data = "DONT CARE"; 14835 const i::byte data[] = {1, 2, 3, 4};
14836 int invalid_size = 3; 14836 int invalid_size = 3;
14837 i::ScriptData* sd = i::ScriptData::New(data, invalid_size); 14837 i::ScriptData* sd = new i::ScriptData(data, invalid_size);
14838 CHECK_EQ(NULL, sd); 14838 i::ParseData* pd = new i::ParseData(sd);
14839 delete sd;
14840 delete pd;
14839 } 14841 }
14840 14842
14841 14843
14842 TEST(CompileWithInvalidCachedData) { 14844 static const char* cached_script_source =
14845 "function foo(){ return 5;}\n"
14846 "function bar(){ return 6 + 7;} foo();";
14847
14848
14849 // ScriptData private implementation details
14850 static const int kHeaderSize = i::PreparseDataConstants::kHeaderSize;
14851 static const int kFunctionEntrySize = i::FunctionEntry::kSize;
14852 static const int kFunctionEntryStartOffset = 0;
14853 static const int kFunctionEntryEndOffset = 1;
14854
14855
14856 TEST(ExpectFailCompileWithInvalidCachedData1) {
14843 v8::V8::Initialize(); 14857 v8::V8::Initialize();
14844 v8::Isolate* isolate = CcTest::isolate(); 14858 v8::Isolate* isolate = CcTest::isolate();
14845 LocalContext context; 14859 LocalContext context;
14846 v8::HandleScope scope(context->GetIsolate()); 14860 v8::HandleScope scope(context->GetIsolate());
14847 i::FLAG_min_preparse_length = 0; 14861 i::FLAG_min_preparse_length = 0;
14848 14862
14849 const char* script = "function foo(){ return 5;}\n" 14863 v8::ScriptCompiler::Source source(v8_str(cached_script_source));
14850 "function bar(){ return 6 + 7;} foo();";
14851 v8::ScriptCompiler::Source source(v8_str(script));
14852 v8::ScriptCompiler::Compile(isolate, &source, 14864 v8::ScriptCompiler::Compile(isolate, &source,
14853 v8::ScriptCompiler::kProduceDataToCache); 14865 v8::ScriptCompiler::kProduceDataToCache);
14854 // source owns its cached data. Create a ScriptData based on it. The user 14866 // source owns its cached data. Create a ScriptData based on it. The user
14855 // never needs to create ScriptDatas any more; we only need it here because we 14867 // never needs to create ScriptDatas any more; we only need it here because we
14856 // want to modify the data before passing it back. 14868 // want to modify the data before passing it back.
14857 const v8::ScriptCompiler::CachedData* cd = source.GetCachedData(); 14869 const v8::ScriptCompiler::CachedData* cd = source.GetCachedData();
14858 // ScriptData does not take ownership of the buffers passed to it. 14870 // ScriptData does not take ownership of the buffers passed to it.
14859 i::ScriptData* sd = 14871 i::ScriptData* sd = new i::ScriptData(cd->data, cd->length);
14860 i::ScriptData::New(reinterpret_cast<const char*>(cd->data), cd->length); 14872 i::ParseData pd(sd);
14861 CHECK(!sd->HasError()); 14873 CHECK(!pd.HasError());
14862 // ScriptData private implementation details 14874 unsigned* pd_data = pd.Data();
14863 const int kHeaderSize = i::PreparseDataConstants::kHeaderSize;
14864 const int kFunctionEntrySize = i::FunctionEntry::kSize;
14865 const int kFunctionEntryStartOffset = 0;
14866 const int kFunctionEntryEndOffset = 1;
14867 unsigned* sd_data =
14868 reinterpret_cast<unsigned*>(const_cast<char*>(sd->Data()));
14869 14875
14870 // Overwrite function bar's end position with 0. 14876 // Overwrite function bar's end position with 0.
14871 sd_data[kHeaderSize + 1 * kFunctionEntrySize + kFunctionEntryEndOffset] = 0; 14877 pd_data[kHeaderSize + 1 * kFunctionEntrySize + kFunctionEntryEndOffset] = 0;
14872 v8::TryCatch try_catch;
14873 14878
14874 // Make the script slightly different so that we don't hit the compilation 14879 // Make the script slightly different so that we don't hit the compilation
14875 // cache. Don't change the lenghts of tokens. 14880 // cache. Don't change the lenghts of tokens.
14876 const char* script2 = "function foo(){ return 6;}\n" 14881 const char* script2 = "function foo(){ return 6;}\n"
14877 "function bar(){ return 6 + 7;} foo();"; 14882 "function bar(){ return 6 + 7;} foo();";
14878 v8::ScriptCompiler::Source source2( 14883 v8::ScriptCompiler::Source source2(
14879 v8_str(script2), 14884 v8_str(script2),
14880 // CachedData doesn't take ownership of the buffers, Source takes 14885 // CachedData doesn't take ownership of the buffers, Source takes
14881 // ownership of CachedData. 14886 // ownership of CachedData.
14882 new v8::ScriptCompiler::CachedData( 14887 new v8::ScriptCompiler::CachedData(sd->data(), sd->length()));
14883 reinterpret_cast<const uint8_t*>(sd->Data()), sd->Length())); 14888 v8::ScriptCompiler::CompileUnbound(isolate, &source2);
14884 Local<v8::UnboundScript> compiled_script =
14885 v8::ScriptCompiler::CompileUnbound(isolate, &source2);
14886 14889
14887 CHECK(try_catch.HasCaught()); 14890 delete sd;
14888 { 14891 }
14889 String::Utf8Value exception_value(try_catch.Message()->Get());
14890 CHECK_EQ("Uncaught SyntaxError: Invalid cached data for function bar",
14891 *exception_value);
14892 }
14893 14892
14894 try_catch.Reset(); 14893
14895 delete sd; 14894 TEST(ExpectFailCompileWithInvalidCachedData2) {
14895 v8::V8::Initialize();
14896 v8::Isolate* isolate = CcTest::isolate();
14897 LocalContext context;
14898 v8::HandleScope scope(context->GetIsolate());
14899 i::FLAG_min_preparse_length = 0;
14900
14901 v8::ScriptCompiler::Source source(v8_str(cached_script_source));
14902 v8::ScriptCompiler::Compile(isolate, &source,
14903 v8::ScriptCompiler::kProduceDataToCache);
14904 // source owns its cached data. Create a ScriptData based on it. The user
14905 // never needs to create ScriptDatas any more; we only need it here because we
14906 // want to modify the data before passing it back.
14907 const v8::ScriptCompiler::CachedData* cd = source.GetCachedData();
14908 // ScriptData does not take ownership of the buffers passed to it.
14909 i::ScriptData* sd = new i::ScriptData(cd->data, cd->length);
14910 i::ParseData pd(sd);
14896 14911
14897 // Overwrite function bar's start position with 200. The function entry will 14912 // Overwrite function bar's start position with 200. The function entry will
14898 // not be found when searching for it by position, and the compilation fails. 14913 // not be found when searching for it by position, and the compilation fails.
14899
14900 // ScriptData does not take ownership of the buffers passed to it. 14914 // ScriptData does not take ownership of the buffers passed to it.
14901 sd = i::ScriptData::New(reinterpret_cast<const char*>(cd->data), cd->length); 14915 unsigned* pd_data = pd.Data();
14902 sd_data = reinterpret_cast<unsigned*>(const_cast<char*>(sd->Data())); 14916 pd_data[kHeaderSize + 1 * kFunctionEntrySize + kFunctionEntryStartOffset] =
14903 sd_data[kHeaderSize + 1 * kFunctionEntrySize + kFunctionEntryStartOffset] =
14904 200; 14917 200;
14905 const char* script3 = "function foo(){ return 7;}\n" 14918 const char* script2 =
14919 "function foo(){ return 7;}\n"
14906 "function bar(){ return 6 + 7;} foo();"; 14920 "function bar(){ return 6 + 7;} foo();";
14907 v8::ScriptCompiler::Source source3( 14921 v8::ScriptCompiler::Source source2(
14908 v8_str(script3), 14922 v8_str(script2),
14909 new v8::ScriptCompiler::CachedData( 14923 new v8::ScriptCompiler::CachedData(sd->data(), sd->length()));
14910 reinterpret_cast<const uint8_t*>(sd->Data()), sd->Length())); 14924 v8::ScriptCompiler::CompileUnbound(isolate, &source2);
14911 compiled_script =
14912 v8::ScriptCompiler::CompileUnbound(isolate, &source3);
14913 CHECK(try_catch.HasCaught());
14914 {
14915 String::Utf8Value exception_value(try_catch.Message()->Get());
14916 CHECK_EQ("Uncaught SyntaxError: Invalid cached data for function bar",
14917 *exception_value);
14918 }
14919 CHECK(compiled_script.IsEmpty());
14920 try_catch.Reset();
14921 delete sd;
14922
14923 // Try passing in cached data which is obviously invalid (wrong length).
14924 sd = i::ScriptData::New(reinterpret_cast<const char*>(cd->data), cd->length);
14925 const char* script4 =
14926 "function foo(){ return 8;}\n"
14927 "function bar(){ return 6 + 7;} foo();";
14928 v8::ScriptCompiler::Source source4(
14929 v8_str(script4),
14930 new v8::ScriptCompiler::CachedData(
14931 reinterpret_cast<const uint8_t*>(sd->Data()), sd->Length() - 1));
14932 compiled_script =
14933 v8::ScriptCompiler::CompileUnbound(isolate, &source4);
14934 CHECK(try_catch.HasCaught());
14935 {
14936 String::Utf8Value exception_value(try_catch.Message()->Get());
14937 CHECK_EQ("Uncaught SyntaxError: Invalid cached data",
14938 *exception_value);
14939 }
14940 CHECK(compiled_script.IsEmpty());
14941 delete sd; 14925 delete sd;
14942 } 14926 }
14943 14927
14928
14929 TEST(ExpectFailCompileWithInvalidCachedData3) {
14930 v8::V8::Initialize();
14931 v8::Isolate* isolate = CcTest::isolate();
14932 LocalContext context;
14933 v8::HandleScope scope(context->GetIsolate());
14934 i::FLAG_min_preparse_length = 0;
14935
14936 v8::ScriptCompiler::Source source(v8_str(cached_script_source));
14937 v8::ScriptCompiler::Compile(isolate, &source,
14938 v8::ScriptCompiler::kProduceDataToCache);
14939 // source owns its cached data. Create a ScriptData based on it. The user
14940 // never needs to create ScriptDatas any more; we only need it here because we
14941 // want to modify the data before passing it back.
14942 const v8::ScriptCompiler::CachedData* cd = source.GetCachedData();
14943 // ScriptData does not take ownership of the buffers passed to it.
14944 i::ScriptData* sd = new i::ScriptData(cd->data, cd->length);
14945
14946 // Try passing in cached data which is obviously invalid (wrong length).
14947 const char* script2 =
14948 "function foo(){ return 8;}\n"
14949 "function bar(){ return 6 + 7;} foo();";
14950 v8::ScriptCompiler::Source source2(
14951 v8_str(script2),
14952 new v8::ScriptCompiler::CachedData(sd->data(), sd->length() - 1));
14953 v8::ScriptCompiler::CompileUnbound(isolate, &source2);
14954 delete sd;
14955 }
14956
14944 14957
14945 // This tests that we do not allow dictionary load/call inline caches 14958 // This tests that we do not allow dictionary load/call inline caches
14946 // to use functions that have not yet been compiled. The potential 14959 // to use functions that have not yet been compiled. The potential
14947 // problem of loading a function that has not yet been compiled can 14960 // problem of loading a function that has not yet been compiled can
14948 // arise because we share code between contexts via the compilation 14961 // arise because we share code between contexts via the compilation
14949 // cache. 14962 // cache.
14950 THREADED_TEST(DictionaryICLoadedFunction) { 14963 THREADED_TEST(DictionaryICLoadedFunction) {
14951 v8::HandleScope scope(CcTest::isolate()); 14964 v8::HandleScope scope(CcTest::isolate());
14952 // Test LoadIC. 14965 // Test LoadIC.
14953 for (int i = 0; i < 2; i++) { 14966 for (int i = 0; i < 2; i++) {
(...skipping 7935 matching lines...) Expand 10 before | Expand all | Expand 10 after
22889 desc = x->GetOwnPropertyDescriptor(v8_str("p1")); 22902 desc = x->GetOwnPropertyDescriptor(v8_str("p1"));
22890 Local<Function> set = 22903 Local<Function> set =
22891 Local<Function>::Cast(Local<Object>::Cast(desc)->Get(v8_str("set"))); 22904 Local<Function>::Cast(Local<Object>::Cast(desc)->Get(v8_str("set")));
22892 Local<Function> get = 22905 Local<Function> get =
22893 Local<Function>::Cast(Local<Object>::Cast(desc)->Get(v8_str("get"))); 22906 Local<Function>::Cast(Local<Object>::Cast(desc)->Get(v8_str("get")));
22894 CHECK_EQ(v8_num(13), get->Call(x, 0, NULL)); 22907 CHECK_EQ(v8_num(13), get->Call(x, 0, NULL));
22895 Handle<Value> args[] = { v8_num(14) }; 22908 Handle<Value> args[] = { v8_num(14) };
22896 set->Call(x, 1, args); 22909 set->Call(x, 1, args);
22897 CHECK_EQ(v8_num(14), get->Call(x, 0, NULL)); 22910 CHECK_EQ(v8_num(14), get->Call(x, 0, NULL));
22898 } 22911 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698