void IPIn::drop(const std::string& msg, const std::string& buf)
{
- log << "IPIn::drop: " << msg << "\n\tBuffer: " << buf << "\n";
+ log << "IPIn::drop (" << ip << "): " << msg << "\n\tBuffer: " << buf
+ << "\n";
}
void IPIn::drop(const std::string& msg, const IPHeader& iph)
{
- log << "IPIn::drop: " << msg << "\n\tIPHeader: " << iph << "\n";
+ log << "IPIn::drop (" << ip << "): " << msg << "\n\tIPHeader: " << iph
+ << "\n";
}
/// Recibe un paquete IP
-std::string IPIn::recv(uint8_t proto, IPAddr& src, IPAddr& dst) throw (std::runtime_error)
+std::string IPIn::recv(uint8_t proto, IPAddr& src, IPAddr& dst, bool filter)
+ throw (std::runtime_error)
{
while (true)
{
+ // Reviso buffers completos a ver si puedo sacar algo
+ buffer_complete_type::iterator f = buffer_complete.find(proto);
+ if (f != buffer_complete.end()) // Hay algo
+ {
+ srcmap_type& srcmap = f->second;
+ srcmap_type::iterator b;
+ if (filter)
+ b = srcmap.find(src); // Si filtro, busco el que me interesa
+ else
+ b = srcmap.begin(); // Si no filtro, agarro el primero
+ if (b != srcmap.end())
+ {
+ src = b->first;
+ dst = (b->second).dst;
+ std::string data = b->second.data;
+ // Limpio buffers
+ srcmap.erase(src);
+ if (srcmap.empty())
+ buffer_complete.erase(proto);
+ return data;
+ }
+ }
+ // Si no pude sacar nada, recibo
std::string buf = dev.receive();
// No es siquiera IP
if (buf.size() < IPHeader::header_len())
{
+ // Silencioso
drop("Cabecera incompleta o no es IP", buf);
continue;
}
IPHeader iph(buf);
#ifdef DEBUG
- std::cout << "IPIn::recv: IPHeader: " << iph << "\n";
+ std::cout << "IPIn::recv (" << ip << "): IPHeader: " << iph << "\n";
std::string tmp = buf.substr(iph.header_len());
std::cout << "\tdata (" << tmp.size() << ") = " << tmp << "\n";
#endif
if (iph.version != 4)
{
+ // Silencioso
drop("Versión IP incorrecta", iph);
continue;
}
if (!iph.check_checksum())
{
+ // Silencioso
drop("Mal checksum", iph);
continue;
}
+ // Si el TTL se va a 0
+ if (!--iph.ttl)
+ {
+ // ICMP
+ drop("TTL == 0 -> ICMP", iph);
+ continue;
+ }
// No es para nosotros y no forwardeamos
if (iph.dst != ip && !forward)
{
+ // Silencioso
drop("No es para nosotros y no hacemos forward", iph);
continue;
}
// Es para nosotros pero somos router
else if (router)
{
+ // Silencioso
drop("Es para nosotros pero somos un router", iph);
continue;
}
// Es para nosotros y somos un host
// Guarda en buffer
buffer[iph][iph.offset] = buf.substr(iph.header_len());
- // Si tiene más fragmentos o es un protocolo distinto, sigo
- if (iph.mf || (iph.proto != proto))
+ // Si tiene más fragmentos, sigo
+ if (iph.mf)
continue;
// No hay más fragmentos, reensamblamos (de ser necesario)
std::string data;
data += i->second;
}
#ifdef DEBUG
- std::cout << "IPIn::recv: Paquete completo: data = '" << data << "'\n";
+ std::cout << "IPIn::recv (" << ip << "): Paquete completo: data = '"
+ << data << "'\n";
#endif
buffer.erase(iph);
+ // Si es un protocolo distinto o no es de quien quiero, rearmo paquete
+ // (ya sé que no hay más fragmentos) y guardo en buffer nomás
+ if ((iph.proto != proto) || (filter == true && src != iph.src))
+ {
+ buffer_complete[proto][iph.src] = BufferPkt(iph.dst, data);
+ continue;
+ }
//TODO faltaría limpiar fragmentos viejos cada tanto (timer?)
src = iph.src;
dst = iph.dst;