|  |  |  | @ -2,99 +2,69 @@ | 
		
	
		
			
				|  |  |  |  | #include <Console.h> | 
		
	
		
			
				|  |  |  |  | #include <Export.h> | 
		
	
		
			
				|  |  |  |  | #include <PluginManager.h> | 
		
	
		
			
				|  |  |  |  | #include <modules/Maps.h> | 
		
	
		
			
				|  |  |  |  | #include <modules/Items.h> | 
		
	
		
			
				|  |  |  |  | #include <modules/Units.h> | 
		
	
		
			
				|  |  |  |  | #include <modules/Gui.h> | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | using namespace DFHack; | 
		
	
		
			
				|  |  |  |  | #include "DataDefs.h" | 
		
	
		
			
				|  |  |  |  | #include "df/world.h" | 
		
	
		
			
				|  |  |  |  | #include "df/map_block.h" | 
		
	
		
			
				|  |  |  |  | #include "df/block_square_event.h" | 
		
	
		
			
				|  |  |  |  | #include "df/block_square_event_material_spatterst.h" | 
		
	
		
			
				|  |  |  |  | #include "df/item_actual.h" | 
		
	
		
			
				|  |  |  |  | #include "df/unit.h" | 
		
	
		
			
				|  |  |  |  | #include "df/matter_state.h" | 
		
	
		
			
				|  |  |  |  | #include "df/cursor.h" | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | #include "modules/Materials.h" | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | #include <vector> | 
		
	
		
			
				|  |  |  |  | #include <string> | 
		
	
		
			
				|  |  |  |  | #include <string.h> | 
		
	
		
			
				|  |  |  |  | using std::vector; | 
		
	
		
			
				|  |  |  |  | using std::string; | 
		
	
		
			
				|  |  |  |  | using namespace DFHack; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result clean (Core * c, vector <string> & parameters); | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result spotclean (Core * c, vector <string> & parameters); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport const char * plugin_name ( void ) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     return "cleaners"; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result plugin_init ( Core * c, std::vector <PluginCommand> &commands) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     commands.clear(); | 
		
	
		
			
				|  |  |  |  |     commands.push_back(PluginCommand("clean","Removes contaminants from map tiles, items and creatures.",clean)); | 
		
	
		
			
				|  |  |  |  |     commands.push_back(PluginCommand("spotclean","Cleans map tile under cursor.",spotclean,cursor_hotkey)); | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result plugin_shutdown ( Core * c ) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | using df::global::world; | 
		
	
		
			
				|  |  |  |  | using df::global::cursor; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | command_result cleanmap (Core * c, bool snow, bool mud) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     const uint32_t water_idx = 6; | 
		
	
		
			
				|  |  |  |  |     const uint32_t mud_idx = 12; | 
		
	
		
			
				|  |  |  |  |     vector<DFHack::t_spattervein *> splatter; | 
		
	
		
			
				|  |  |  |  |     DFHack::Maps *Mapz = c->getMaps(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     // init the map
 | 
		
	
		
			
				|  |  |  |  |     if(!Mapz->Start()) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         c->con << "Can't init map." << std::endl; | 
		
	
		
			
				|  |  |  |  |         c->Resume(); | 
		
	
		
			
				|  |  |  |  |         return CR_FAILURE; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     uint32_t x_max,y_max,z_max; | 
		
	
		
			
				|  |  |  |  |     Mapz->getSize(x_max,y_max,z_max); | 
		
	
		
			
				|  |  |  |  |     int num_blocks = 0; | 
		
	
		
			
				|  |  |  |  |     int blocks_total = 0; | 
		
	
		
			
				|  |  |  |  |     // walk the map
 | 
		
	
		
			
				|  |  |  |  |     for(uint32_t x = 0; x< x_max;x++) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         for(uint32_t y = 0; y< y_max;y++) | 
		
	
		
			
				|  |  |  |  |         { | 
		
	
		
			
				|  |  |  |  |             for(uint32_t z = 0; z< z_max;z++) | 
		
	
		
			
				|  |  |  |  |     // Invoked from clean(), already suspended
 | 
		
	
		
			
				|  |  |  |  |     int num_blocks = 0, blocks_total = world->map.map_blocks.size(); | 
		
	
		
			
				|  |  |  |  |     for (int i = 0; i < blocks_total; i++) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |                 df_block * block = Mapz->getBlock(x,y,z); | 
		
	
		
			
				|  |  |  |  |                 if(block) | 
		
	
		
			
				|  |  |  |  |                 { | 
		
	
		
			
				|  |  |  |  |                     blocks_total ++; | 
		
	
		
			
				|  |  |  |  |         df::map_block *block = world->map.map_blocks[i]; | 
		
	
		
			
				|  |  |  |  |         bool cleaned = false; | 
		
	
		
			
				|  |  |  |  |                     Mapz->SortBlockEvents(x,y,z,0,0,&splatter); | 
		
	
		
			
				|  |  |  |  |                     for(int i = 0; i < 16; i++) | 
		
	
		
			
				|  |  |  |  |                         for(int j = 0; j < 16; j++) | 
		
	
		
			
				|  |  |  |  |         for(int x = 0; x < 16; x++) | 
		
	
		
			
				|  |  |  |  |         { | 
		
	
		
			
				|  |  |  |  |             for(int y = 0; y < 16; y++) | 
		
	
		
			
				|  |  |  |  |             { | 
		
	
		
			
				|  |  |  |  |                             block->occupancy[i][j].bits.arrow_color = 0; | 
		
	
		
			
				|  |  |  |  |                             block->occupancy[i][j].bits.broken_arrows_variant = 0; | 
		
	
		
			
				|  |  |  |  |                 block->occupancy[x][y].bits.arrow_color = 0; | 
		
	
		
			
				|  |  |  |  |                 block->occupancy[x][y].bits.arrow_variant = 0; | 
		
	
		
			
				|  |  |  |  |             } | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  |                     for(uint32_t i = 0; i < splatter.size(); i++) | 
		
	
		
			
				|  |  |  |  |         for (int j = 0; j < block->block_events.size(); j++) | 
		
	
		
			
				|  |  |  |  |         { | 
		
	
		
			
				|  |  |  |  |                         DFHack::t_spattervein * vein = splatter[i]; | 
		
	
		
			
				|  |  |  |  |             df::block_square_event *evt = block->block_events[j]; | 
		
	
		
			
				|  |  |  |  |             if (evt->getType() != df::block_square_event_type::material_spatter) | 
		
	
		
			
				|  |  |  |  |                 continue; | 
		
	
		
			
				|  |  |  |  |             // type verified - recast to subclass
 | 
		
	
		
			
				|  |  |  |  |             df::block_square_event_material_spatterst *spatter = (df::block_square_event_material_spatterst *)evt; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |             // filter snow
 | 
		
	
		
			
				|  |  |  |  |             if(!snow | 
		
	
		
			
				|  |  |  |  |                             && vein->mat1 == DFHack::Materials::WATER | 
		
	
		
			
				|  |  |  |  |                             && vein->matter_state == DFHack::state_powder) | 
		
	
		
			
				|  |  |  |  |                 && spatter->mat_type == DFHack::Materials::WATER | 
		
	
		
			
				|  |  |  |  |                 && spatter->mat_state == df::matter_state::Powder) | 
		
	
		
			
				|  |  |  |  |                 continue; | 
		
	
		
			
				|  |  |  |  |             // filter mud
 | 
		
	
		
			
				|  |  |  |  |             if(!mud | 
		
	
		
			
				|  |  |  |  |                             && vein->mat1 == DFHack::Materials::MUD | 
		
	
		
			
				|  |  |  |  |                             && vein->matter_state == DFHack::state_solid) | 
		
	
		
			
				|  |  |  |  |                 && spatter->mat_type == DFHack::Materials::MUD | 
		
	
		
			
				|  |  |  |  |                 && spatter->mat_state == df::matter_state::Solid) | 
		
	
		
			
				|  |  |  |  |                 continue; | 
		
	
		
			
				|  |  |  |  |                         Mapz->RemoveBlockEvent(x,y,z,(t_virtual *) vein); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |             delete evt; | 
		
	
		
			
				|  |  |  |  |             block->block_events.erase(block->block_events.begin() + j); | 
		
	
		
			
				|  |  |  |  |             j--; | 
		
	
		
			
				|  |  |  |  |             cleaned = true; | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  |         num_blocks += cleaned; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |             } | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     if(num_blocks) | 
		
	
		
			
				|  |  |  |  |         c->con.print("Cleaned %d of %d map blocks.\n", num_blocks, blocks_total); | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
	
		
			
				
					|  |  |  | @ -102,30 +72,19 @@ command_result cleanmap (Core * c, bool snow, bool mud) | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | command_result cleanitems (Core * c) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     DFHack::Items * Items = c->getItems(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     vector <df_item*> p_items; | 
		
	
		
			
				|  |  |  |  |     if(!Items->readItemVector(p_items)) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         c->con.printerr("Can't access the item vector.\n"); | 
		
	
		
			
				|  |  |  |  |         c->Resume(); | 
		
	
		
			
				|  |  |  |  |         return CR_FAILURE; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     std::size_t numItems = p_items.size(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     // Invoked from clean(), already suspended
 | 
		
	
		
			
				|  |  |  |  |     int cleaned_items = 0, cleaned_total = 0; | 
		
	
		
			
				|  |  |  |  |     for (std::size_t i = 0; i < numItems; i++) | 
		
	
		
			
				|  |  |  |  |     for (int i = 0; i < world->items.all.size(); i++) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         df_item * itm = p_items[i]; | 
		
	
		
			
				|  |  |  |  |         if(!itm->contaminants) | 
		
	
		
			
				|  |  |  |  |             continue; | 
		
	
		
			
				|  |  |  |  |         if (itm->contaminants->size()) | 
		
	
		
			
				|  |  |  |  |         // currently, all item classes extend item_actual, so this should be safe
 | 
		
	
		
			
				|  |  |  |  |         df::item_actual *item = (df::item_actual *)world->items.all[i]; | 
		
	
		
			
				|  |  |  |  |         if (item->contaminants && item->contaminants->size()) | 
		
	
		
			
				|  |  |  |  |         { | 
		
	
		
			
				|  |  |  |  |             for(int j = 0; j < itm->contaminants->size(); j++) | 
		
	
		
			
				|  |  |  |  |                 delete itm->contaminants->at(j); | 
		
	
		
			
				|  |  |  |  |             for (int j = 0; j < item->contaminants->size(); j++) | 
		
	
		
			
				|  |  |  |  |                 delete item->contaminants->at(j); | 
		
	
		
			
				|  |  |  |  |             cleaned_items++; | 
		
	
		
			
				|  |  |  |  |             cleaned_total += itm->contaminants->size(); | 
		
	
		
			
				|  |  |  |  |             itm->contaminants->clear(); | 
		
	
		
			
				|  |  |  |  |             cleaned_total += item->contaminants->size(); | 
		
	
		
			
				|  |  |  |  |             item->contaminants->clear(); | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     if (cleaned_total) | 
		
	
	
		
			
				
					|  |  |  | @ -135,27 +94,19 @@ command_result cleanitems (Core * c) | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | command_result cleanunits (Core * c) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     DFHack::Units * Creatures = c->getUnits(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     uint32_t num_creatures; | 
		
	
		
			
				|  |  |  |  |     if (!Creatures->Start(num_creatures)) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         c->con.printerr("Can't read unit list!\n"); | 
		
	
		
			
				|  |  |  |  |         c->Resume(); | 
		
	
		
			
				|  |  |  |  |         return CR_FAILURE; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     // Invoked from clean(), already suspended
 | 
		
	
		
			
				|  |  |  |  |     int num_units = world->units.all.size(); | 
		
	
		
			
				|  |  |  |  |     int cleaned_units = 0, cleaned_total = 0; | 
		
	
		
			
				|  |  |  |  |     for (std::size_t i = 0; i < num_creatures; i++) | 
		
	
		
			
				|  |  |  |  |     for (int i = 0; i < num_units; i++) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         df_unit *unit = Creatures->creatures->at(i); | 
		
	
		
			
				|  |  |  |  |         int num = unit->contaminants.size(); | 
		
	
		
			
				|  |  |  |  |         if (num) | 
		
	
		
			
				|  |  |  |  |         df::unit *unit = world->units.all[i]; | 
		
	
		
			
				|  |  |  |  |         if (unit->body.spatters.size()) | 
		
	
		
			
				|  |  |  |  |         { | 
		
	
		
			
				|  |  |  |  |             for(int j = 0; j < unit->contaminants.size(); j++) | 
		
	
		
			
				|  |  |  |  |                 delete unit->contaminants.at(j); | 
		
	
		
			
				|  |  |  |  |             for (int j = 0; j < unit->body.spatters.size(); j++) | 
		
	
		
			
				|  |  |  |  |                 delete unit->body.spatters[j]; | 
		
	
		
			
				|  |  |  |  |             cleaned_units++; | 
		
	
		
			
				|  |  |  |  |             cleaned_total += num; | 
		
	
		
			
				|  |  |  |  |             unit->contaminants.clear(); | 
		
	
		
			
				|  |  |  |  |             cleaned_total += unit->body.spatters.size(); | 
		
	
		
			
				|  |  |  |  |             unit->body.spatters.clear(); | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     if (cleaned_total) | 
		
	
	
		
			
				
					|  |  |  | @ -163,33 +114,40 @@ command_result cleanunits (Core * c) | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | // This is slightly different from what's in the Maps module - it takes tile coordinates rather than block coordinates
 | 
		
	
		
			
				|  |  |  |  | df::map_block *getBlock (int32_t x, int32_t y, int32_t z) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     if ((x < 0) || (y < 0) || (z < 0)) | 
		
	
		
			
				|  |  |  |  |         return NULL; | 
		
	
		
			
				|  |  |  |  |     if ((x >= world->map.x_count) || (y >= world->map.y_count) || (z >= world->map.z_count)) | 
		
	
		
			
				|  |  |  |  |         return NULL; | 
		
	
		
			
				|  |  |  |  |     // block_index isn't declared correctly - needs one more level of indirection
 | 
		
	
		
			
				|  |  |  |  |     return ((df::map_block ****)world->map.block_index)[x >> 4][y >> 4][z]; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result spotclean (Core * c, vector <string> & parameters) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     // HOTKEY COMMAND: CORE ALREADY SUSPENDED
 | 
		
	
		
			
				|  |  |  |  |     vector<DFHack::t_spattervein *> splatter; | 
		
	
		
			
				|  |  |  |  |     DFHack::Maps *Mapz = c->getMaps(); | 
		
	
		
			
				|  |  |  |  |     DFHack::Gui *Gui = c->getGui(); | 
		
	
		
			
				|  |  |  |  |     // init the map
 | 
		
	
		
			
				|  |  |  |  |     if(!Mapz->Start()) | 
		
	
		
			
				|  |  |  |  |     if (cursor->x == -30000) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         c->con.printerr("Can't init map.\n"); | 
		
	
		
			
				|  |  |  |  |         c->con.printerr("The cursor is not active.\n"); | 
		
	
		
			
				|  |  |  |  |         return CR_FAILURE; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     int32_t cursorX, cursorY, cursorZ; | 
		
	
		
			
				|  |  |  |  |     Gui->getCursorCoords(cursorX,cursorY,cursorZ); | 
		
	
		
			
				|  |  |  |  |     if(cursorX == -30000) | 
		
	
		
			
				|  |  |  |  |     df::map_block *block = getBlock(cursor->x, cursor->y, cursor->z); | 
		
	
		
			
				|  |  |  |  |     if (block == NULL) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         c->con.printerr("The cursor is not active.\n"); | 
		
	
		
			
				|  |  |  |  |         c->con.printerr("Invalid map block selected!\n"); | 
		
	
		
			
				|  |  |  |  |         return CR_FAILURE; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     int32_t blockX = cursorX / 16, blockY = cursorY / 16; | 
		
	
		
			
				|  |  |  |  |     int32_t tileX = cursorX % 16, tileY = cursorY % 16; | 
		
	
		
			
				|  |  |  |  |     df_block *b = Mapz->getBlock(blockX,blockY,cursorZ); | 
		
	
		
			
				|  |  |  |  |     vector <t_spattervein *> spatters; | 
		
	
		
			
				|  |  |  |  |     Mapz->SortBlockEvents(blockX, blockY, cursorZ, 0,0, &spatters); | 
		
	
		
			
				|  |  |  |  |     for(int i = 0; i < spatters.size(); i++) | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     for (int i = 0; i < block->block_events.size(); i++) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |         spatters[i]->intensity[tileX][tileY] = 0; | 
		
	
		
			
				|  |  |  |  |         df::block_square_event *evt = block->block_events[i]; | 
		
	
		
			
				|  |  |  |  |         if (evt->getType() != df::block_square_event_type::material_spatter) | 
		
	
		
			
				|  |  |  |  |             continue; | 
		
	
		
			
				|  |  |  |  |         // type verified - recast to subclass
 | 
		
	
		
			
				|  |  |  |  |         df::block_square_event_material_spatterst *spatter = (df::block_square_event_material_spatterst *)evt; | 
		
	
		
			
				|  |  |  |  |         spatter->amount[cursor->x % 16][cursor->y % 16] = 0; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
	
		
			
				
					|  |  |  | @ -253,3 +211,21 @@ DFhackCExport command_result clean (Core * c, vector <string> & parameters) | 
		
	
		
			
				|  |  |  |  |     c->Resume(); | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport const char * plugin_name ( void ) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     return "cleaners"; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result plugin_init ( Core * c, std::vector <PluginCommand> &commands) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     commands.clear(); | 
		
	
		
			
				|  |  |  |  |     commands.push_back(PluginCommand("clean","Removes contaminants from map tiles, items and creatures.",clean)); | 
		
	
		
			
				|  |  |  |  |     commands.push_back(PluginCommand("spotclean","Cleans map tile under cursor.",spotclean,cursor_hotkey)); | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | DFhackCExport command_result plugin_shutdown ( Core * c ) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     return CR_OK; | 
		
	
		
			
				|  |  |  |  | } |