// Copyright (c) Hercules Dev Team, licensed under GNU GPL. // See the LICENSE file // Portions Copyright (c) Athena Dev Teams #include "../common/cbasetypes.h" #include "../common/socket.h" #include "../common/timer.h" #include "../common/malloc.h" #include "../common/nullpo.h" #include "../common/showmsg.h" #include "../common/strlib.h" #include "../common/utils.h" #include "../common/db.h" #include "clif.h" #include "instance.h" #include "map.h" #include "npc.h" #include "party.h" #include "pc.h" #include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdarg.h> #include <time.h> struct instance_interface instance_s; /// Checks whether given instance id is valid or not. bool instance_is_valid(int instance_id) { if( instance_id < 0 || instance_id >= instance->instances ) {// out of range return false; } if( instance->list[instance_id].state == INSTANCE_FREE ) {// uninitialized/freed instance slot return false; } return true; } /*-------------------------------------- * name : instance name * Return value could be * -4 = already exists | -3 = no free instances | -2 = owner not found | -1 = invalid type * On success return instance_id *--------------------------------------*/ int instance_create(int owner_id, const char *name, enum instance_owner_type type) { struct map_session_data *sd = NULL; unsigned short *icptr = NULL; struct party_data *p = NULL; struct guild *g = NULL; short *iptr = NULL; int i, j; switch ( type ) { case IOT_NONE: break; case IOT_CHAR: if( ( sd = map->id2sd(owner_id) ) == NULL ) { ShowError("instance_create: character %d not found for instance '%s'.\n", owner_id, name); return -2; } iptr = sd->instance; icptr = &sd->instances; break; case IOT_PARTY: if( ( p = party->search(owner_id) ) == NULL ) { ShowError("instance_create: party %d not found for instance '%s'.\n", owner_id, name); return -2; } iptr = p->instance; icptr = &p->instances; break; case IOT_GUILD: if( ( g = guild->search(owner_id) ) == NULL ) { ShowError("instance_create: guild %d not found for instance '%s'.\n", owner_id, name); return -2; } iptr = g->instance; icptr = &g->instances; break; default: ShowError("instance_create: unknown type %d for owner_id %d and name %s.\n", type,owner_id,name); return -1; } if( type != IOT_NONE && *icptr ) { ARR_FIND(0, *icptr, i, strcmp(instance->list[iptr[i]].name,name) == 0 ); if( i != *icptr ) return -4;/* already got this instance */ } ARR_FIND(0, instance->instances, i, instance->list[i].state == INSTANCE_FREE); if( i == instance->instances ) RECREATE(instance->list, struct instance_data, ++instance->instances); instance->list[i].state = INSTANCE_IDLE; instance->list[i].id = i; instance->list[i].idle_timer = INVALID_TIMER; instance->list[i].idle_timeout = instance->list[i].idle_timeoutval = 0; instance->list[i].progress_timer = INVALID_TIMER; instance->list[i].progress_timeout = 0; instance->list[i].users = 0; instance->list[i].map = NULL; instance->list[i].num_map = 0; instance->list[i].owner_id = owner_id; instance->list[i].owner_type = type; instance->list[i].vars = idb_alloc(DB_OPT_RELEASE_DATA); safestrncpy( instance->list[i].name, name, sizeof(instance->list[i].name) ); if( type != IOT_NONE ) { ARR_FIND(0, *icptr, j, iptr[j] == -1); if( j == *icptr ) { switch( type ) { case IOT_CHAR: RECREATE(sd->instance, short, ++*icptr); sd->instance[sd->instances-1] = i; break; case IOT_PARTY: RECREATE(p->instance, short, ++*icptr); p->instance[p->instances-1] = i; break; case IOT_GUILD: RECREATE(g->instance, short, ++*icptr); g->instance[g->instances-1] = i; break; } } else iptr[j] = i; } clif->instance(i, 1, 0); // Start instancing window return i; } /*-------------------------------------- * Add a map to the instance using src map "name" *--------------------------------------*/ int instance_add_map(const char *name, int instance_id, bool usebasename, const char *map_name) { int16 m = map->mapname2mapid(name); int i, im = -1; size_t num_cell, size; if( m < 0 ) return -1; // source map not found if( !instance->valid(instance_id) ) { ShowError("instance_add_map: trying to attach '%s' map to non-existing instance %d.\n", name, instance_id); return -1; } if( map_name != NULL && strdb_iget(mapindex_db, map_name) ) { ShowError("instance_add_map: trying to create instanced map with existent name '%s'\n", map_name); return -2; } if( maplist[m].instance_id >= 0 ) { // Source map already belong to a Instance. ShowError("instance_add_map: trying to instance already instanced map %s.\n", name); return -4; } ARR_FIND( instance->start_id, map->map_num, i, maplist[i].name[0] == 0 ); // Searching for a Free Map if( i < map->map_num ) im = i; // Unused map found (old instance) else { im = map->map_num; // Using next map index RECREATE(maplist,struct map_data,++map->map_num); } if( maplist[m].cell == (struct mapcell *)0xdeadbeaf ) map->cellfromcache(&maplist[m]); memcpy( &maplist[im], &maplist[m], sizeof(struct map_data) ); // Copy source map if( map_name != NULL ) { snprintf(maplist[im].name, MAP_NAME_LENGTH, "%s", map_name); maplist[im].custom_name = true; } else snprintf(maplist[im].name, MAP_NAME_LENGTH, (usebasename ? "%.3d#%s" : "%.3d%s"), instance_id, name); // Generate Name for Instance Map maplist[im].index = mapindex_addmap(-1, maplist[im].name); // Add map index maplist[im].channel = NULL; if( !maplist[im].index ) { maplist[im].name[0] = '\0'; ShowError("instance_add_map: no more free map indexes.\n"); return -3; // No free map index } // Reallocate cells num_cell = maplist[im].xs * maplist[im].ys; CREATE( maplist[im].cell, struct mapcell, num_cell ); memcpy( maplist[im].cell, maplist[m].cell, num_cell * sizeof(struct mapcell) ); size = maplist[im].bxs * maplist[im].bys * sizeof(struct block_list*); maplist[im].block = (struct block_list**)aCalloc(size, 1); maplist[im].block_mob = (struct block_list**)aCalloc(size, 1); memset(maplist[im].npc, 0x00, sizeof(maplist[i].npc)); maplist[im].npc_num = 0; memset(maplist[im].moblist, 0x00, sizeof(maplist[im].moblist)); maplist[im].mob_delete_timer = INVALID_TIMER; //Mimic unit if( maplist[m].unit_count ) { maplist[im].unit_count = maplist[m].unit_count; CREATE( maplist[im].units, struct mapflag_skill_adjust*, maplist[im].unit_count ); for(i = 0; i < maplist[im].unit_count; i++) { CREATE( maplist[im].units[i], struct mapflag_skill_adjust, 1); memcpy( maplist[im].units[i],maplist[m].units[i],sizeof(struct mapflag_skill_adjust)); } } //Mimic skills if( maplist[m].skill_count ) { maplist[im].skill_count = maplist[m].skill_count; CREATE( maplist[im].skills, struct mapflag_skill_adjust*, maplist[im].skill_count ); for(i = 0; i < maplist[im].skill_count; i++) { CREATE( maplist[im].skills[i], struct mapflag_skill_adjust, 1); memcpy( maplist[im].skills[i],maplist[m].skills[i],sizeof(struct mapflag_skill_adjust)); } } //Mimic zone mf if( maplist[m].zone_mf_count ) { maplist[im].zone_mf_count = maplist[m].zone_mf_count; CREATE( maplist[im].zone_mf, char *, maplist[im].zone_mf_count ); for(i = 0; i < maplist[im].zone_mf_count; i++) { CREATE(maplist[im].zone_mf[i], char, MAP_ZONE_MAPFLAG_LENGTH); safestrncpy(maplist[im].zone_mf[i],maplist[m].zone_mf[i],MAP_ZONE_MAPFLAG_LENGTH); } } maplist[im].m = im; maplist[im].instance_id = instance_id; maplist[im].instance_src_map = m; maplist[m].flag.src4instance = 1; // Flag this map as a src map for instances RECREATE(instance->list[instance_id].map, unsigned short, ++instance->list[instance_id].num_map); instance->list[instance_id].map[instance->list[instance_id].num_map - 1] = im; // Attach to actual instance map->addmap2db(&maplist[im]); return im; } /*-------------------------------------- * m : source map of this instance * party_id : source party of this instance * type : result (0 = map id | 1 = instance id) *--------------------------------------*/ int instance_map2imap(int16 m, int instance_id) { int i; if( !instance->valid(instance_id) ) { return -1; } for( i = 0; i < instance->list[instance_id].num_map; i++ ) { if( instance->list[instance_id].map[i] && maplist[instance->list[instance_id].map[i]].instance_src_map == m ) return instance->list[instance_id].map[i]; } return -1; } /*-------------------------------------- * m : source map * instance_id : where to search * result : mapid of map "m" in this instance *--------------------------------------*/ int instance_mapid2imapid(int16 m, int instance_id) { if( maplist[m].flag.src4instance == 0 ) return m; // not instances found for this map else if( maplist[m].instance_id >= 0 ) { // This map is a instance, not a src map instance ShowError("map_instance_mapid2imapid: already instanced (%d / %d)\n", m, instance_id); return -1; } if( !instance->valid(instance_id) ) return -1; return instance->map2imap(m, instance_id); } /*-------------------------------------- * map_instance_map_npcsub * Used on Init instance. Duplicates each script on source map *--------------------------------------*/ int instance_map_npcsub(struct block_list* bl, va_list args) { struct npc_data* nd = (struct npc_data*)bl; int16 m = va_arg(args, int); // Destination Map if ( npc->duplicate4instance(nd, m) ) ShowDebug("instance_map_npcsub:npc_duplicate4instance failed (%s/%d)\n",nd->name,m); return 1; } /*-------------------------------------- * Init all map on the instance. Npcs are created here *--------------------------------------*/ void instance_init(int instance_id) { int i; if( !instance->valid(instance_id) ) return; // nothing to do for( i = 0; i < instance->list[instance_id].num_map; i++ ) map->foreachinmap(instance_map_npcsub, maplist[instance->list[instance_id].map[i]].instance_src_map, BL_NPC, instance->list[instance_id].map[i]); instance->list[instance_id].state = INSTANCE_BUSY; } /*-------------------------------------- * Used on instance deleting process. * Warps all players on each instance map to its save points. *--------------------------------------*/ int instance_del_load(struct map_session_data* sd, va_list args) { int16 m = va_arg(args,int); if( !sd || sd->bl.m != m ) return 0; pc->setpos(sd, sd->status.save_point.map, sd->status.save_point.x, sd->status.save_point.y, CLR_OUTSIGHT); return 1; } /* for npcs behave differently when being unloaded within a instance */ int instance_cleanup_sub(struct block_list *bl, va_list ap) { nullpo_ret(bl); switch(bl->type) { case BL_PC: map->quit((struct map_session_data *) bl); break; case BL_NPC: npc->unload((struct npc_data *)bl,true); break; case BL_MOB: unit->free(bl,CLR_OUTSIGHT); break; case BL_PET: //There is no need for this, the pet is removed together with the player. [Skotlex] break; case BL_ITEM: map->clearflooritem(bl); break; case BL_SKILL: skill->delunit((struct skill_unit *) bl); break; } return 1; } /*-------------------------------------- * Removes a simple instance map *--------------------------------------*/ void instance_del_map(int16 m) { int i; if( m <= 0 || maplist[m].instance_id == -1 ) { ShowError("instance_del_map: tried to remove non-existing instance map (%d)\n", m); return; } map->map_foreachpc(instance_del_load, m); map->foreachinmap(instance_cleanup_sub, m, BL_ALL); if( maplist[m].mob_delete_timer != INVALID_TIMER ) timer->delete(maplist[m].mob_delete_timer, map->removemobs_timer); mapindex_removemap(map_id2index(m)); // Free memory aFree(maplist[m].cell); aFree(maplist[m].block); aFree(maplist[m].block_mob); if( maplist[m].unit_count ) { for(i = 0; i < maplist[m].unit_count; i++) { aFree(maplist[m].units[i]); } if( maplist[m].units ) aFree(maplist[m].units); } if( maplist[m].skill_count ) { for(i = 0; i < maplist[m].skill_count; i++) { aFree(maplist[m].skills[i]); } if( maplist[m].skills ) aFree(maplist[m].skills); } if( maplist[m].zone_mf_count ) { for(i = 0; i < maplist[m].zone_mf_count; i++) { aFree(maplist[m].zone_mf[i]); } if( maplist[m].zone_mf ) aFree(maplist[m].zone_mf); } // Remove from instance for( i = 0; i < instance->list[maplist[m].instance_id].num_map; i++ ) { if( instance->list[maplist[m].instance_id].map[i] == m ) { instance->list[maplist[m].instance_id].num_map--; for( ; i < instance->list[maplist[m].instance_id].num_map; i++ ) instance->list[maplist[m].instance_id].map[i] = instance->list[maplist[m].instance_id].map[i+1]; i = -1; break; } } if( i == instance->list[maplist[m].instance_id].num_map ) ShowError("map_instance_del: failed to remove %s from instance list (%s): %d\n", maplist[m].name, instance->list[maplist[m].instance_id].name, m); if( maplist[m].channel ) clif->chsys_delete(maplist[m].channel); map->removemapdb(&maplist[m]); memset(&maplist[m], 0x00, sizeof(maplist[0])); maplist[m].name[0] = 0; maplist[m].instance_id = -1; maplist[m].mob_delete_timer = INVALID_TIMER; } /*-------------------------------------- * Timer to destroy instance by process or idle *--------------------------------------*/ int instance_destroy_timer(int tid, unsigned int tick, int id, intptr_t data) { instance->destroy(id); return 0; } /*-------------------------------------- * Removes a instance, all its maps and npcs. *--------------------------------------*/ void instance_destroy(int instance_id) { struct map_session_data *sd = NULL; unsigned short *icptr = NULL; struct party_data *p = NULL; struct guild *g = NULL; short *iptr = NULL; int type, j, last = 0; unsigned int now = (unsigned int)time(NULL); if( !instance->valid(instance_id) ) return; // nothing to do if( instance->list[instance_id].progress_timeout && instance->list[instance_id].progress_timeout <= now ) type = 1; else if( instance->list[instance_id].idle_timeout && instance->list[instance_id].idle_timeout <= now ) type = 2; else type = 3; clif->instance(instance_id, 5, type); // Report users this instance has been destroyed switch ( instance->list[instance_id].owner_type ) { case IOT_NONE: break; case IOT_CHAR: if( ( sd = map->id2sd(instance->list[instance_id].owner_id) ) == NULL ) { break; } iptr = sd->instance; icptr = &sd->instances; break; case IOT_PARTY: if( ( p = party->search(instance->list[instance_id].owner_id) ) == NULL ) { break; } iptr = p->instance; icptr = &p->instances; break; case IOT_GUILD: if( ( g = guild->search(instance->list[instance_id].owner_id) ) == NULL ) { break; } iptr = g->instance; icptr = &g->instances; break; default: ShowError("instance_destroy: unknown type %d for owner_id %d and name '%s'.\n", instance->list[instance_id].owner_type,instance->list[instance_id].owner_id,instance->list[instance_id].name); break; } if( iptr != NULL ) { ARR_FIND(0, *icptr, j, iptr[j] == instance_id); if( j != *icptr ) iptr[j] = -1; } while( instance->list[instance_id].num_map && last != instance->list[instance_id].map[0] ) { // Remove all maps from instance last = instance->list[instance_id].map[0]; instance->del_map( instance->list[instance_id].map[0] ); } if( instance->list[instance_id].vars ) db_destroy(instance->list[instance_id].vars); if( instance->list[instance_id].progress_timer != INVALID_TIMER ) timer->delete( instance->list[instance_id].progress_timer, instance->destroy_timer); if( instance->list[instance_id].idle_timer != INVALID_TIMER ) timer->delete( instance->list[instance_id].idle_timer, instance->destroy_timer); instance->list[instance_id].vars = NULL; if( instance->list[instance_id].map ) aFree(instance->list[instance_id].map); instance->list[instance_id].map = NULL; instance->list[instance_id].state = INSTANCE_FREE; instance->list[instance_id].num_map = 0; } /*-------------------------------------- * Checks if there are users in the instance or not to start idle timer *--------------------------------------*/ void instance_check_idle(int instance_id) { bool idle = true; unsigned int now = (unsigned int)time(NULL); if( !instance->valid(instance_id) || instance->list[instance_id].idle_timeoutval == 0 ) return; if( instance->list[instance_id].users ) idle = false; if( instance->list[instance_id].idle_timer != INVALID_TIMER && !idle ) { timer->delete(instance->list[instance_id].idle_timer, instance->destroy_timer); instance->list[instance_id].idle_timer = INVALID_TIMER; instance->list[instance_id].idle_timeout = 0; clif->instance(instance_id, 3, 0); // Notify instance users normal instance expiration } else if( instance->list[instance_id].idle_timer == INVALID_TIMER && idle ) { instance->list[instance_id].idle_timeout = now + instance->list[instance_id].idle_timeoutval; instance->list[instance_id].idle_timer = timer->add( timer->gettick() + instance->list[instance_id].idle_timeoutval * 1000, instance->destroy_timer, instance_id, 0); clif->instance(instance_id, 4, 0); // Notify instance users it will be destroyed of no user join it again in "X" time } } /*-------------------------------------- * Set instance Timers *--------------------------------------*/ void instance_set_timeout(int instance_id, unsigned int progress_timeout, unsigned int idle_timeout) { unsigned int now = (unsigned int)time(0); if( !instance->valid(instance_id) ) return; if( instance->list[instance_id].progress_timer != INVALID_TIMER ) timer->delete( instance->list[instance_id].progress_timer, instance->destroy_timer); if( instance->list[instance_id].idle_timer != INVALID_TIMER ) timer->delete( instance->list[instance_id].idle_timer, instance->destroy_timer); if( progress_timeout ) { instance->list[instance_id].progress_timeout = now + progress_timeout; instance->list[instance_id].progress_timer = timer->add( timer->gettick() + progress_timeout * 1000, instance->destroy_timer, instance_id, 0); } else { instance->list[instance_id].progress_timeout = 0; instance->list[instance_id].progress_timer = INVALID_TIMER; } if( idle_timeout ) { instance->list[instance_id].idle_timeoutval = idle_timeout; instance->list[instance_id].idle_timer = INVALID_TIMER; instance->check_idle(instance_id); } else { instance->list[instance_id].idle_timeoutval = 0; instance->list[instance_id].idle_timeout = 0; instance->list[instance_id].idle_timer = INVALID_TIMER; } if( instance->list[instance_id].idle_timer == INVALID_TIMER && instance->list[instance_id].progress_timer != INVALID_TIMER ) clif->instance(instance_id, 3, 0); } /*-------------------------------------- * Checks if sd in on a instance and should be kicked from it *--------------------------------------*/ void instance_check_kick(struct map_session_data *sd) { int16 m = sd->bl.m; clif->instance_leave(sd->fd); if( maplist[m].instance_id >= 0 ) { // User was on the instance map if( maplist[m].save.map ) pc->setpos(sd, maplist[m].save.map, maplist[m].save.x, maplist[m].save.y, CLR_TELEPORT); else pc->setpos(sd, sd->status.save_point.map, sd->status.save_point.x, sd->status.save_point.y, CLR_TELEPORT); } } void do_final_instance(void) { int i; for(i = 0; i < instance->instances; i++) { instance->destroy(i); } if( instance->list ) aFree(instance->list); instance->list = NULL; instance->instances = 0; } void do_init_instance(void) { timer->add_func_list(instance->destroy_timer, "instance_destroy_timer"); } void instance_defaults(void) { instance = &instance_s; instance->init = do_init_instance; instance->final = do_final_instance; /* start point */ instance->start_id = 0; /* count */ instance->instances = 0; /* */ instance->list = NULL; /* */ instance->create = instance_create; instance->add_map = instance_add_map; instance->del_map = instance_del_map; instance->map2imap = instance_map2imap; instance->mapid2imapid = instance_mapid2imapid; instance->destroy = instance_destroy; instance->start = instance_init; instance->check_idle = instance_check_idle; instance->check_kick = instance_check_kick; instance->set_timeout = instance_set_timeout; instance->valid = instance_is_valid; instance->destroy_timer = instance_destroy_timer; }