void free_client(struct s_client *cl) { if (!cl) return; struct s_reader *rdr = cl->reader; // Remove client from client list. kill_thread also removes this client, so here just if client exits itself... struct s_client *prev, *cl2; cs_writelock(&clientlist_lock); cl->kill = 1; for (prev = first_client, cl2 = first_client->next; prev->next != NULL; prev = prev->next, cl2 = cl2->next) { if (cl == cl2) break; } if (cl == cl2) prev->next = cl2->next; // Remove client from list int32_t bucket = (uintptr_t)cl / 16 % CS_CLIENT_HASHBUCKETS; // Remove client from hashed list if (first_client_hashed[bucket] == cl) { first_client_hashed[bucket] = cl->nexthashed; } else { for (prev = first_client_hashed[bucket], cl2 = first_client_hashed[bucket]->nexthashed; prev->nexthashed != NULL; prev = prev->nexthashed, cl2 = cl2->nexthashed) { if (cl == cl2) break; } if (cl == cl2) prev->nexthashed = cl2->nexthashed; } cs_writeunlock(&clientlist_lock); // Clean reader. The cleaned structures should be only used by the reader thread, so we should be save without waiting if (rdr) { remove_reader_from_ecm(rdr); remove_reader_from_active(rdr); if(rdr->ph.cleanup) rdr->ph.cleanup(cl); if (cl->typ == 'r') cardreader_close(rdr); if (cl->typ == 'p') network_tcp_connection_close(rdr, "cleanup"); cl->reader = NULL; } // Clean client specific data if (cl->typ == 'c') { cs_statistics(cl); cl->last_caid = 0xFFFF; cl->last_srvid = 0xFFFF; cs_statistics(cl); cs_sleepms(500); //just wait a bit that really really nobody is accessing client data struct s_module *module = get_module(cl); if (module->cleanup) module->cleanup(cl); } // Close network socket if not already cleaned by previous cleanup functions if (cl->pfd) close(cl->pfd); // Clean all remaining structures free_joblist(cl); cleanup_ecmtasks(cl); add_garbage(cl->emmcache); #ifdef MODULE_CCCAM add_garbage(cl->cc); #endif #ifdef MODULE_SERIAL add_garbage(cl->serialdata); #endif add_garbage(cl); }
bool cardreader_init(struct s_reader *reader) { struct s_client *client = reader->client; client->typ = 'r'; int8_t i = 0; set_localhost_ip(&client->ip); while((cardreader_device_init(reader) == 2) && i < 10) { cs_sleepms(2000); if(!ll_contains(configured_readers, reader) || !is_valid_client(client) || reader->enable != 1) { return false; } i++; } if (i >= 10) { reader->card_status = READER_DEVICE_ERROR; cardreader_close(reader); reader->enable = 0; return false; } else { if(reader->typ == R_INTERNAL) { if(boxtype_is("dm8000") || boxtype_is("dm800") || boxtype_is("dm800se")) {reader->cardmhz = 2700;} if(boxtype_is("dm500") || boxtype_is("dm600pvr")) {reader->cardmhz = 3150;} if(boxtype_is("dm7025")) {reader->cardmhz = 8300;} if((!strncmp(boxtype_get(), "vu", 2 ))||(boxtype_is("ini-8000am"))) {reader->cardmhz = 2700; reader->mhz = 450;} // only one speed for vu+ and Atemio Nemesis due to usage of TDA8024 } if((reader->cardmhz > 2000) && (reader->typ != R_SMART)) { rdr_log(reader, "Reader initialized (device=%s, detect=%s%s, pll max=%.2f MHz, wanted mhz=%.2f MHz)", reader->device, reader->detect & 0x80 ? "!" : "", RDR_CD_TXT[reader->detect & 0x7f], (float)reader->cardmhz / 100, (float)reader->mhz / 100); rdr_log(reader,"Reader sci internal, detected box type: %s", boxtype_get()); } else { if (reader->typ == R_SMART || is_smargo_reader(reader)) { rdr_log_dbg(reader, D_IFD, "clocking for smartreader with smartreader or smargo protocol"); if (reader->cardmhz >= 2000) reader->cardmhz = 369; else if (reader->cardmhz >= 1600) reader->cardmhz = 1600; else if (reader->cardmhz >= 1200) reader->cardmhz = 1200; else if (reader->cardmhz >= 961) reader->cardmhz = 961; else if (reader->cardmhz >= 800) reader->cardmhz = 800; else if (reader->cardmhz >= 686) reader->cardmhz = 686; else if (reader->cardmhz >= 600) reader->cardmhz = 600; else if (reader->cardmhz >= 534) reader->cardmhz = 534; else if (reader->cardmhz >= 480) reader->cardmhz = 480; else if (reader->cardmhz >= 436) reader->cardmhz = 436; else if (reader->cardmhz >= 400) reader->cardmhz = 400; else if (reader->cardmhz >= 369) reader->cardmhz = 369; else if (reader->cardmhz == 357) reader->cardmhz = 369; else // 357 not a default smartreader setting if (reader->cardmhz >= 343) reader->cardmhz = 343; else reader->cardmhz = 320; if (reader->mhz >= 1600) reader->mhz = 1600; else if (reader->mhz >= 1200) reader->mhz = 1200; else if (reader->mhz >= 961) reader->mhz = 961; else if (reader->mhz >= 900) reader->mhz = 900; else if (reader->mhz >= 800) reader->mhz = 800; else if (reader->mhz >= 686) reader->mhz = 686; else if (reader->mhz >= 600) reader->mhz = 600; else if (reader->mhz >= 534) reader->mhz = 534; else if (reader->mhz >= 480) reader->mhz = 480; else if (reader->mhz >= 436) reader->mhz = 436; else if (reader->mhz >= 400) reader->mhz = 369; else if (reader->mhz >= 369) reader->mhz = 369; else if (reader->mhz == 357) reader->mhz = 369; else // 357 not a default smartreader setting if (reader->mhz >= 343) reader->mhz = 343; else reader->mhz = 320; } if ((reader->typ == R_SMART || is_smargo_reader(reader)) && reader->autospeed == 1) { rdr_log(reader, "Reader initialized (device=%s, detect=%s%s, mhz= AUTO, cardmhz=%d)", reader->device, reader->detect & 0x80 ? "!" : "", RDR_CD_TXT[reader->detect & 0x7f], reader->cardmhz); } else { rdr_log(reader, "Reader initialized (device=%s, detect=%s%s, mhz=%d, cardmhz=%d)", reader->device, reader->detect & 0x80 ? "!" : "", RDR_CD_TXT[reader->detect & 0x7f], reader->mhz, reader->cardmhz); if (reader->typ == R_INTERNAL && !(reader->cardmhz > 2000)) rdr_log(reader,"Reader sci internal, detected box type: %s", boxtype_get()); } } return true; } }
bool cardreader_init(struct s_reader *reader) { struct s_client *client = reader->client; client->typ = 'r'; int8_t i = 0; set_localhost_ip(&client->ip); while((cardreader_device_init(reader) == 2) && i < 10) { cs_sleepms(2000); if(!ll_contains(configured_readers, reader) || !is_valid_client(client) || reader->enable != 1) { return false; } i++; } if (i >= 10) { reader->card_status = READER_DEVICE_ERROR; cardreader_close(reader); reader->enable = 0; return false; } else { if((reader->cardmhz > 2000) && (reader->typ != R_SMART)) { rdr_log(reader, "Reader initialized (device=%s, detect=%s%s, pll max=%.2f MHz, wanted mhz=%.2f MHz)", reader->device, reader->detect & 0x80 ? "!" : "", RDR_CD_TXT[reader->detect & 0x7f], (float)reader->cardmhz / 100, (float)reader->mhz / 100); rdr_log(reader,"Reader sci internal, detected box type: %s", stb_boxtype ? stb_boxtype : "generic"); } else { if ((reader->typ == R_SMART) || (!strcasecmp(reader->crdr.desc, "smargo")) ){ rdr_debug_mask(reader, D_IFD, "clocking for smartreader with smartreader or smargo protocol"); if (reader->cardmhz >= 2000) reader->cardmhz = 369; else if (reader->cardmhz >= 1600) reader->cardmhz = 1600; else if (reader->cardmhz >= 1200) reader->cardmhz = 1200; else if (reader->cardmhz >= 961) reader->cardmhz = 961; else if (reader->cardmhz >= 800) reader->cardmhz = 800; else if (reader->cardmhz >= 686) reader->cardmhz = 686; else if (reader->cardmhz >= 600) reader->cardmhz = 600; else if (reader->cardmhz >= 534) reader->cardmhz = 534; else if (reader->cardmhz >= 480) reader->cardmhz = 480; else if (reader->cardmhz >= 436) reader->cardmhz = 436; else if (reader->cardmhz >= 400) reader->cardmhz = 400; else if (reader->cardmhz >= 369) reader->cardmhz = 369; else if (reader->cardmhz == 357) reader->cardmhz = 369; else // 357 not a default smartreader setting if (reader->cardmhz >= 343) reader->cardmhz = 343; else reader->cardmhz = 320; if (reader->mhz >= 1600) reader->mhz = 1600; else if (reader->mhz >= 1200) reader->mhz = 1200; else if (reader->mhz >= 961) reader->mhz = 961; else if (reader->mhz >= 900) reader->mhz = 900; else if (reader->mhz >= 800) reader->mhz = 800; else if (reader->mhz >= 686) reader->mhz = 686; else if (reader->mhz >= 600) reader->mhz = 600; else if (reader->mhz >= 534) reader->mhz = 534; else if (reader->mhz >= 480) reader->mhz = 480; else if (reader->mhz >= 436) reader->mhz = 436; else if (reader->mhz >= 400) reader->mhz = 369; else if (reader->mhz >= 369) reader->mhz = 369; else if (reader->mhz == 357) reader->mhz = 369; else // 357 not a default smartreader setting if (reader->mhz >= 343) reader->mhz = 343; else reader->mhz = 320; } if (((reader->typ == R_SMART) && (reader->autospeed == 1)) || ((!strcasecmp(reader->crdr.desc, "smargo")) && (reader->autospeed == 1))) { rdr_log(reader, "Reader initialized (device=%s, detect=%s%s, mhz= AUTO, cardmhz=%d)", reader->device, reader->detect & 0x80 ? "!" : "", RDR_CD_TXT[reader->detect & 0x7f], reader->cardmhz); } else { rdr_log(reader, "Reader initialized (device=%s, detect=%s%s, mhz=%d, cardmhz=%d)", reader->device, reader->detect & 0x80 ? "!" : "", RDR_CD_TXT[reader->detect & 0x7f], reader->mhz, reader->cardmhz); } } return true; } }
void free_client(struct s_client *cl) { if(!cl) { return; } struct s_reader *rdr = cl->reader; // Remove client from client list. kill_thread also removes this client, so here just if client exits itself... struct s_client *prev, *cl2; cs_writelock(__func__, &clientlist_lock); if(!cl->kill_started) { cl->kill_started = 1; } else { cs_writeunlock(__func__, &clientlist_lock); cs_log("[free_client] ERROR: free already started!"); return; } cl->kill = 1; for(prev = first_client, cl2 = first_client->next; prev->next != NULL; prev = prev->next, cl2 = cl2->next) { if(cl == cl2) { break; } } if(cl == cl2) { prev->next = cl2->next; } // Remove client from list int32_t bucket = (uintptr_t)cl / 16 % CS_CLIENT_HASHBUCKETS; // Remove client from hashed list if(first_client_hashed[bucket] == cl) { first_client_hashed[bucket] = cl->nexthashed; } else { for(prev = first_client_hashed[bucket], cl2 = first_client_hashed[bucket]->nexthashed; prev->nexthashed != NULL; prev = prev->nexthashed, cl2 = cl2->nexthashed) { if(cl == cl2) { break; } } if(cl == cl2) { prev->nexthashed = cl2->nexthashed; } } cs_writeunlock(__func__, &clientlist_lock); cleanup_ecmtasks(cl); // Clean reader. The cleaned structures should be only used by the reader thread, so we should be save without waiting if(rdr) { ll_destroy_data(&rdr->emmstat); remove_reader_from_active(rdr); cs_sleepms(1000); //just wait a bit that really really nobody is accessing client data if(rdr->ph.cleanup) { rdr->ph.cleanup(cl); } if(cl->typ == 'r') { cardreader_close(rdr); } if(cl->typ == 'p') { network_tcp_connection_close(rdr, "cleanup"); } cl->reader = NULL; } // Clean client specific data if(cl->typ == 'c') { cs_statistics(cl); cl->last_caid = NO_CAID_VALUE; cl->last_provid = NO_PROVID_VALUE; cl->last_srvid = NO_SRVID_VALUE; cs_statistics(cl); cs_sleepms(1000); //just wait a bit that really really nobody is accessing client data } struct s_module *module = get_module(cl); if(module->cleanup) { module->cleanup(cl); } // Close network socket if not already cleaned by previous cleanup functions if(cl->pfd) { close(cl->pfd); } // Clean all remaining structures free_joblist(cl); NULLFREE(cl->work_mbuf); if(cl->ecmtask) { add_garbage(cl->ecmtask); cl->ecmtask = NULL; } ll_destroy_data(&cl->cascadeusers); ftab_clear(&cl->ftab); ftab_clear(&cl->fchid); tuntab_clear(&cl->ttab); caidtab_clear(&cl->ctab); NULLFREE(cl->cltab.aclass); NULLFREE(cl->cltab.bclass); NULLFREE(cl->cw_rass); ll_destroy_data(&cl->ra_buf); NULLFREE(cl->aes_keys); #ifdef MODULE_CCCAM add_garbage(cl->cc); #endif #ifdef MODULE_SERIAL add_garbage(cl->serialdata); #endif add_garbage(cl); }