151 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			151 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifdef PUGIXML_COMPACT
 | |
| #include "test.hpp"
 | |
| 
 | |
| using namespace pugi;
 | |
| 
 | |
| static void overflow_hash_table(xml_document& doc)
 | |
| {
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 
 | |
| 	// compact encoding assumes next_sibling is a forward-only pointer so we can allocate hash entries by reordering nodes
 | |
| 	// we allocate enough hash entries to be exactly on the edge of rehash threshold
 | |
| 	for (int i = 0; i < 8; ++i)
 | |
| 		CHECK(n.prepend_child(node_element));
 | |
| }
 | |
| 
 | |
| TEST_XML_FLAGS(compact_out_of_memory_string, "<n a='v'/><?n v?>", parse_pi)
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_attribute a = doc.child(STR("n")).attribute(STR("a"));
 | |
| 	xml_node pi = doc.last_child();
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!pi.set_name(STR("name"))));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!pi.set_value(STR("value"))));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!a.set_name(STR("name"))));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!a.set_value(STR("value"))));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_out_of_memory_attribute, "<n a='v'/>")
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 	xml_attribute a = n.attribute(STR("a"));
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.append_attribute(STR(""))));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.prepend_attribute(STR(""))));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.insert_attribute_after(STR(""), a)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.insert_attribute_before(STR(""), a)));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_out_of_memory_attribute_copy, "<n a='v'/>")
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 	xml_attribute a = n.attribute(STR("a"));
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.append_copy(a)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.prepend_copy(a)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.insert_copy_after(a, a)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.insert_copy_before(a, a)));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_out_of_memory_node, "<n/>")
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.append_child(node_element)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.prepend_child(node_element)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.insert_child_after(node_element, n)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.insert_child_before(node_element, n)));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_out_of_memory_node_copy, "<n/>")
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.append_copy(n)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.prepend_copy(n)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.insert_copy_after(n, n)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.insert_copy_before(n, n)));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_out_of_memory_node_move, "<n/><ne/>")
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 	xml_node ne = doc.child(STR("ne"));
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.append_move(n)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.prepend_move(n)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.insert_move_after(n, ne)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.insert_move_before(n, ne)));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_out_of_memory_remove, "<n a='v'/>")
 | |
| {
 | |
| 	test_runner::_memory_fail_threshold = 1;
 | |
| 
 | |
| 	overflow_hash_table(doc);
 | |
| 
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 	xml_attribute a = n.attribute(STR("a"));
 | |
| 
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!n.remove_attribute(a)));
 | |
| 	CHECK_ALLOC_FAIL(CHECK(!doc.remove_child(n)));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_pointer_attribute_list, "<n a='v'/>")
 | |
| {
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 	xml_attribute a = n.attribute(STR("a"));
 | |
| 
 | |
| 	// make sure we fill the page with node x
 | |
| 	for (int i = 0; i < 1000; ++i)
 | |
| 		doc.append_child(STR("x"));
 | |
| 
 | |
| 	// this requires extended encoding for prev_attribute_c/next_attribute
 | |
| 	n.append_attribute(STR("b"));
 | |
| 
 | |
| 	// this requires extended encoding for first_attribute
 | |
| 	n.remove_attribute(a);
 | |
| 
 | |
| 	CHECK(!n.attribute(STR("a")));
 | |
| 	CHECK(n.attribute(STR("b")));
 | |
| }
 | |
| 
 | |
| TEST_XML(compact_pointer_node_list, "<n/>")
 | |
| {
 | |
| 	xml_node n = doc.child(STR("n"));
 | |
| 
 | |
| 	// make sure we fill the page with node x
 | |
| 	// this requires extended encoding for prev_sibling_c/next_sibling
 | |
| 	for (int i = 0; i < 1000; ++i)
 | |
| 		doc.append_child(STR("x"));
 | |
| 
 | |
| 	// this requires extended encoding for first_child
 | |
| 	n.append_child(STR("child"));
 | |
| 
 | |
| 	CHECK(n.child(STR("child")));
 | |
| }
 | |
| #endif
 |