171 lines
5.1 KiB
C
Executable File
171 lines
5.1 KiB
C
Executable File
#define MODULE_LOG_PREFIX "constcw"
|
|
|
|
//FIXME Not checked on threadsafety yet; after checking please remove this line
|
|
#include "globals.h"
|
|
#ifdef MODULE_CONSTCW
|
|
#include "oscam-client.h"
|
|
#include "oscam-ecm.h"
|
|
#include "oscam-net.h"
|
|
#include "oscam-string.h"
|
|
|
|
static int32_t pserver;
|
|
|
|
int32_t constcw_file_available(void)
|
|
{
|
|
FILE *fp;
|
|
|
|
fp = fopen(cur_client()->reader->device, "r");
|
|
if(!fp)
|
|
{
|
|
cs_log("ERROR: Can't open %s (errno=%d %s)", cur_client()->reader->device, errno, strerror(errno));
|
|
return (0);
|
|
}
|
|
fclose(fp);
|
|
return (1);
|
|
}
|
|
|
|
int32_t constcw_analyse_file(uint16_t c_caid, uint32_t c_prid, uint16_t c_sid, uint16_t c_pmtpid, uint32_t c_vpid, uint16_t c_ecmpid, uint8_t *dcw)
|
|
{
|
|
//CAID:PROVIDER:SID:PMTPID:ECMPID:VPID:XX XX XX XX XX XX XX XX XX XX XX XX XX XX XX XX
|
|
FILE *fp;
|
|
char token[512];
|
|
uint32_t caid, provid, sid, vpid, pmtpid, ecmpid;
|
|
int32_t cw[16];
|
|
|
|
fp = fopen(cur_client()->reader->device, "r");
|
|
if(!fp)
|
|
{
|
|
cs_log("ERROR: Can't open %s (errno=%d %s)", cur_client()->reader->device, errno, strerror(errno));
|
|
return (0);
|
|
}
|
|
|
|
cs_log("Searching CW for CAID %04X PROVID %06X SRVID %04X ECMPID %04X PMTPID %04X VPID %04X", c_caid, c_prid, c_sid, c_ecmpid, c_pmtpid, c_vpid);
|
|
|
|
while(fgets(token, sizeof(token), fp))
|
|
{
|
|
if(token[0] == '#') { continue; }
|
|
vpid = 0;
|
|
int ret = sscanf(token, "%4x:%6x:%4x:%4x:%4x::%2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x", &caid, &provid, &sid, &pmtpid, &ecmpid,
|
|
&cw[0], &cw[1], &cw[2], &cw[3], &cw[4], &cw[5], &cw[6], &cw[7],
|
|
&cw[8], &cw[9], &cw[10], &cw[11], &cw[12], &cw[13], &cw[14], &cw[15]);
|
|
|
|
if(ret != 21){
|
|
ret = sscanf(token, "%4x:%6x:%4x:%4x:%4x:%4x:%2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x %2x", &caid, &provid, &sid, &pmtpid, &ecmpid, &vpid,
|
|
&cw[0], &cw[1], &cw[2], &cw[3], &cw[4], &cw[5], &cw[6], &cw[7],
|
|
&cw[8], &cw[9], &cw[10], &cw[11], &cw[12], &cw[13], &cw[14], &cw[15]);
|
|
if(ret != 22) continue;
|
|
}
|
|
|
|
//cs_log("Line found: %s", token);
|
|
if(c_caid == caid && c_sid == sid && (!provid || provid == c_prid) && (!pmtpid || !c_pmtpid || pmtpid == c_pmtpid) && (!vpid || !c_vpid || vpid == c_vpid)
|
|
&& (!ecmpid || !c_ecmpid || ecmpid == c_ecmpid))
|
|
{
|
|
fclose(fp);
|
|
int8_t i;
|
|
for(i = 0; i < 16; ++i)
|
|
{ dcw[i] = (uint8_t) cw[i]; }
|
|
cs_log("Entry found: %04X@%06X:%04X:%04X:%04X:%04X:%s", caid, provid, sid, pmtpid, ecmpid, vpid, cs_hexdump(1, dcw, 16, token, sizeof(token)));
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
//************************************************************************************************************************
|
|
//* client/server common functions
|
|
//************************************************************************************************************************
|
|
static int32_t constcw_recv(struct s_client *client, uint8_t *buf, int32_t l)
|
|
{
|
|
int32_t ret;
|
|
|
|
if(!client->udp_fd) { return (-9); }
|
|
ret = read(client->udp_fd, buf, l);
|
|
if(ret < 1) { return (-1); }
|
|
client->last = time(NULL);
|
|
return (ret);
|
|
}
|
|
|
|
//************************************************************************************************************************
|
|
//* client functions
|
|
//************************************************************************************************************************
|
|
int32_t constcw_client_init(struct s_client *client)
|
|
{
|
|
int32_t fdp[2];
|
|
|
|
client->pfd = 0;
|
|
if(socketpair(PF_LOCAL, SOCK_STREAM, 0, fdp))
|
|
{
|
|
cs_log("ERROR: Socket creation failed: %s", strerror(errno));
|
|
return 1;
|
|
}
|
|
client->udp_fd = fdp[0];
|
|
pserver = fdp[1];
|
|
|
|
memset((char *) &client->udp_sa, 0, sizeof(client->udp_sa));
|
|
SIN_GET_FAMILY(client->udp_sa) = AF_INET;
|
|
|
|
// Oscam has no reader.au in s_reader like ki's mpcs ;)
|
|
// reader[ridx].au = 0;
|
|
// cs_log("local reader: %s (file: %s) constant cw au=0", reader[ridx].label, reader[ridx].device);
|
|
cs_log("Local reader: %s (file: %s)", client->reader->label, client->reader->device);
|
|
|
|
client->pfd = client->udp_fd;
|
|
|
|
if(constcw_file_available())
|
|
{
|
|
client->reader->tcp_connected = 2;
|
|
client->reader->card_status = CARD_INSERTED;
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int32_t constcw_send_ecm(struct s_client *client, ECM_REQUEST *er)
|
|
{
|
|
time_t t;
|
|
struct s_reader *rdr = client->reader;
|
|
uint8_t cw[16];
|
|
|
|
t = time(NULL);
|
|
// Check if DCW exist in the files
|
|
//cs_log("Searching ConstCW for ECM: %04X@%06X:%04X (%d)", er->caid, er->prid, er->srvid, er->l);
|
|
|
|
if(constcw_analyse_file(er->caid, er->prid, er->srvid, er->pmtpid, er->vpid, er->pid, cw) == 0)
|
|
{
|
|
write_ecm_answer(rdr, er, E_NOTFOUND, (E1_READER << 4 | E2_SID), NULL, NULL, 0, NULL);
|
|
}
|
|
else
|
|
{
|
|
write_ecm_answer(rdr, er, E_FOUND, 0, cw, NULL, 0, NULL);
|
|
}
|
|
|
|
client->last = t;
|
|
rdr->last_g = t;
|
|
return (0);
|
|
}
|
|
|
|
static int32_t constcw_recv_chk(struct s_client *UNUSED(client), uint8_t *UNUSED(dcw), int32_t *rc, uint8_t *UNUSED(buf), int32_t UNUSED(n))
|
|
{
|
|
//dcw = dcw;
|
|
//n = n;
|
|
//buf = buf;
|
|
|
|
*rc = 0;
|
|
return (-1);
|
|
}
|
|
|
|
void module_constcw(struct s_module *ph)
|
|
{
|
|
ph->desc = "constcw";
|
|
ph->type = MOD_NO_CONN;
|
|
ph->listenertype = LIS_CONSTCW;
|
|
ph->recv = constcw_recv;
|
|
|
|
ph->c_init = constcw_client_init;
|
|
ph->c_recv_chk = constcw_recv_chk;
|
|
ph->c_send_ecm = constcw_send_ecm;
|
|
ph->num = R_CONSTCW;
|
|
}
|
|
#endif
|